react-native-easy-app 详解与使用之(一) AsyncStorage

react-native-easy-app 是一款为React Native App快速开发提供基础服务的纯JS库(支持 IOS & Android),特别是在从0到1的项目搭建初期,至少能够为开发者减小30%的工做量。react

react-native-easy-app 主要作了这些工做:
1. 对AsyncStorage进行封装,开发者只需几行代码便可实现一个持久化数据管理器。
2. 对fetch进行封装,使得开发者只需关注当前App的先后台交互逻辑和协议,定义好参数设置及解析逻辑便可。
3. 从新封装了RN的View、Text、Image、FlatList 使用得这些控件在适当的时候支持事件或支持icon与文本,能有效减小布局中的嵌套逻辑。
4. 经过设置一个屏幕参考尺寸,重置XView、XText、XImage的尺寸,实现自动多屏适配git

可能有人以为,RN的AsyncStorage自己就很简单,本身封装也就几十行代码的工做量,为何还要使用第三方库?github

一千我的心中,有一千个哈姆雷特,也许个人封装思路能给你带来不同的启发也未可知呢?数据库

数据存储(AsyncStorage)

RN平台提供的AsyncStorage有一些基础方法:setItem,getItem,removeItem,getAllKeys,这些是promise模式的而且AsyncStorage只支持对纯字符串的存取,所以咱们不便于直接在代码中去直接调用这些方法,咱们得对AsyncStorage作一次封装,怎样封装能使咱们更方便快捷的访问本地存取呢?npm

咱们来看下经过 react-native-easy-app 库的XStorage,咱们能够怎样访问AsyncStorage:react-native

一、核心代码实现
import { XStorage } from 'react-native-easy-app';
import { AsyncStorage } from 'react-native';

let RNStorage = { // 自定义对象
    hasLogin: undefined,
    customerId: undefined,
    userInfo: undefined
};
XStorage.initStorage(RNStorage, AsyncStorage, () => { // 初始化自定义数据管理器
    RNStorage.customerId = '123456';
    RNStorage.hasLogin = true;
    RNStorage.userInfo = {name: 'zhangsan', age: 30};
    console.log(JSON.stringify(RNStorage)) // 打印数据管理器的内容
})

执行完上面的代码后咱们看看控制台输出:
react-native-easy-app 详解与使用之(一) AsyncStorage数组

再经过Root Explorer 查看一下当前App的data/data/{package}/database 下数据表的内容:
react-native-easy-app 详解与使用之(一) AsyncStoragepromise

什么?上面的代码中并无作任何数据库的存储操做啊,为何赋值给RNStorage的数据却被存到了本地数据库中呢?咱们先看上面的代码中作了什么:app

  1. 定义了一个自定义对象RNStorage
  2. 将自定义对象传给XStorage.initStorage 进行初始化
  3. 在初始化完成后对RNStorage的属性进行了赋值
  4. 打印RNStorage的内容

因而可知,数据的存储操做一定是上面的第二、3步引发的。咱们进入XStorage的源码看看,里面作了什么:异步

  • 源代码 1
    Object.keys(targetObj).map(key => {
    const keyStr = newKey(Tag, key);
    Object.defineProperty(targetObj, key, {
        get: () => {
            return this[keyStr]
        },
        set: (value) => {
            try {
                this[keyStr] = value;
                const valueStr = (typeof value === 'object') ? JSON.stringify(value) : String(value);
                keyValuesPairs.push([keyStr, valueStr])
            } catch (exception) {
                console.log(exception && exception.message);
            }
        },
    })
    });
    setInterval(() => {
    if (!isEmpty(keyValuesPairs)) {
        let saveDataArray = [...keyValuesPairs];
        keyValuesPairs = []; //清空原键值对数组
        AsyncStorage.multiSet(saveDataArray, () => {
            dataChangedCallback && dataChangedCallback(saveDataArray)
        });
    }
    }, 2500)
    • 源代码 2
      const Keys = Object.keys(storageObj);
      const StorageKeys = Keys.map(key => newKey(Tag, key));
      // 初始化时,将AsyncStorage中的数据一次性读取到内存中
      AsyncStorage.multiGet(StorageKeys).then(keyValuePairs => {
      keyValuePairs.map(([keyStr, value]) => {
      let [, key] = keyStr.split(splitTag);
      if (persistTag !== key && !isEmpty(value)) {
          storageObj[key] = convertData(value)
      }
      });
      setTimeout(() => initializedCallback(), 100)
      }).catch(error => {
      console.log(error)
      })

      哦,原来 XStorage 经过getter、setter生成器,将用户自定义的 RNStorage 的各属与 AsyncStorage 的数据表各字段的值进行了关联造成了一个绑定关系,在当用户对 RNStorage 的各属性进行赋值、取值操做的时候,实际上会触发getter、setter生成器,相应的会对 AsyncStorage 中的数据表进行读写操做。

效率与性能的平衡

  • <读> 在初始化XStorage的时候就将AsyncStorage中的全部字段一次性读取到 RNStorage 对象中,之后续读取属性时,并不须要通过AsyncStorage,而是直接返回 RNStorage的属性。
  • <写> 在开发者修改XStorage的属性值时,会先将目标数据赋值给XStorage的属性,而后再异步经过AsyncStorage将目标数据写入到数据库中(考虑到数据写入的效率与性能问题,目前的处理方式为:每次数值的变动都会记录下来,定时程序每隔2.5秒进行一次数据批量写入操做),但这个丝绝不会影响App对数据的操做,由于RNStorage中的数据是实时且同步的。

至此就彻底清楚了,是否是很简单?开发者经过 react-native-easy-app 只需定义一个全局可导出的 RNStorage对象(命名随意,并定义好App所需的各属性字段),而后在App启动的时候经过XStorage初始化一次RNStorage便可,之后直接访问RNStorage中的属性值就好了(全部对RNStorage属性的修改都会被自动同步到AsyncStorage中),彻底是一劳永逸啊。。。

react-native-easy-app 详解与使用之(二) fetch

想进一步了解,请移步至 npm 或github查看 react-native-easy-app,有源码及使用示例,待你们一探究竟,欢迎朋友们 Star!

相关文章
相关标签/搜索