FastKV:一个真的很快的KV存储组件

一、前言

KV存储无论对于客户端还是服务端都是重要的构件。
对于Android客户端而言,最常见的莫过于SDK提供的SharePreferences(以下简称SP),但其低效率和ANR问题饱受诟病。
18年年末微信开源了MMKV,写入速度比前者高不少。
后来官方又推出了基于Kotlin的DataStore,测试下来,发现写入效率很低。
我之前写过一个叫LightKV的存储组件,不过当时认知不足,设计不够成熟。

1.1 SP的不足

关于SP的缺点网上有不少讨论,这里主要提两个点:

  • 保存速度较慢

SP用内存层用HashMap保存,磁盘层则是用的XML文件保存。
每次更改,都需要将整个HashMap序列化为XML格式的报文然后整个写入文件。
归结其较慢的原因:
1、不能增量写入;
2、序列化比较耗时。

  • 可以能会导致ANR
public void apply() {
    // ...省略无关代码...
    QueuedWork.addFinisher(awaitCommit);
    Runnable postWriteRunnable = new Runnable() {
        @Override
        public void run() {
            awaitCommit.run();
            QueuedWork.removeFinisher(awaitCommit);
        }
    };
    SharedPreferencesImpl.this.enqueueDiskWrite(mcr, postWriteRunnable);
}
public void handleStopActivity(IBinder token, boolean show, int configChanges,
                               PendingTransactionActions pendingActions, boolean finalStateRequest, String reason) {
    // ...省略无关代码...
    // Make sure any pending writes are now committed.
    if (!r.isPreHoneycomb()) {
        QueuedWork.waitToFinish();
    }
}

Activity stop时会等待SP的写入任务,如果SP的写入任务多且执行慢的话,可能会阻塞主线程较长时间,轻则卡顿,重则ANR。

1.2 MMKV的不足

  • 没有类型信息,不支持getAll
    MMKV的存储用类似于Protobuf的编码方式,只存储key和value本身,没有存类型信息(Protobuf用tag标记字段,信息更少)。
    由于没有记录类型信息,MMKV无法自动反序列化,也就无法实现getAll接口。
  • 读取相对较慢
    SP在加载的时候已经将value反序列化存在HashMap中了,读取的时候索引到之后就能直接引用了。
    而MMKV每次读取时都需要重新解码,除了时间上的消耗之外,还需要每次都创建新的对象。
    不过这不是大问题,相对SP没有差很多。
  • 需要引入so, 增加包体积
    引入MMKV需要增加的体积还是不少的,且不说jar包和aidl文件,光是一个arm64-v8a的so就有四百多K。
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值