资深安卓研发大佬详解MMKV:谷歌都推荐使用的轻量级存储方案

  • 读写方式:I/O

  • 数据格式:xml

  • 写入方式:全量更新

即每当需要更新一项数据,SharedPreferences的整个读写过程都是:将**「所有数据」**转化成xml格式 -> 通过I/O方式写入

2. 容易导致ANR


主要是由于同步提交(commit)、异步提交(Apply) 和 获取数据getXX()导致的。

/*

    1. 同步提交commit
  • commit提交是同步的,直到磁盘操作成功后才会完成

  • 所以当数据量比较大时,使用commit很可能引起ANR

*/

public boolean commit() {

MemoryCommitResult mcr = commitToMemory();

SharedPreferencesImpl.this.enqueueDiskWrite(mcr, null);

try {

mcr.writtenToDiskLatch.await();

} catch (InterruptedException e) {

return false;

}

/*

  • 回调的时机:

    1. commit是在内存和硬盘操作均结束时回调
    1. apply是内存操作结束时就进行回调

*/

notifyListeners(mcr);

return mcr.writeToDiskResult;

}

/*

    1. 异步提交apply
  • 当数据量比较大时,使用apply也可能引起ANR

*/

public void apply() {

final long startTime = System.currentTimeMillis();

final MemoryCommitResult mcr = commitToMemory();

final Runnable awaitCommit = new Runnable() {

@Override

public void run() {

// 启用等待

mcr.writtenToDiskLatch.await();

}

};

// 将 awaitCommit 添加到队列 QueuedWork 中

QueuedWork.addFinisher(awaitCommit);

Runnable postWriteRunnable = new Runnable() {

@Override

public void run() {

awaitCommit.run();

QueuedWork.removeFinisher(awaitCommit);

}

};

// 将写入任务加入到队列中,而写入任务在一个线程中执行

SharedPreferencesImpl.this.enqueueDiskWrite(mcr, postWriteRunnable);

// 为了保证异步任务及时完成,当生命周期处于 handleStopService() 、handlePauseActivity() 、handleStopActivity()时会调用QueuedWork.waitToFinish() 会等待写入任务执行完毕

// waitToFinish() :会一直等待写入任务执行完毕,其它什么都不做。

// 当有很多写入任务,会依次执行;当文件很大时,效率很低,则容易造成 ANR

public static void waitToFinish() {

Runnable toFinish;

while ((toFinish = sPendingWorkFinishers.poll()) != null) {

toFinish.run();

}

/*

    1. 获取数据getXX()
  • 所有 getXXX() 方法都是同步的,在主线程调用 get 方法,必须等待 SP 加载完毕,也有可能导致ANR

*/

// 使用getSharedPreferences() 最终会调用SharedPreferencesImpl#startLoadFromDisk()开启一个线程异步读取数据

new Thread(“SharedPreferencesImpl-load”) {

public void run() {

loadFromDisk();

}

}.start();

// 当我们正在读取一个比较大的数据,还没读取完,接着调用 getXXX()。

public String getString(String key, @Nullable String defValue) {

synchronized (mLock) {

awaitLoadedLocked();

String v = (String)mMap.get(key);

return v != null ? v : defValue;

}

}

// 在同步方法内调用了wait(),会一直等待 getSharedPreferences() 开启的线程读取完数据才能继续往下执行

// 如果读取一个大的文件,也很大可能会造成ANR

private void awaitLoadedLocked() {

while (!mLoaded) {

try {

mLock.wait();

} catch (InterruptedException unused) {

}

}

}

}


MMKV原理

==================================================================

1. 读写方式:内存映射MMAP


MMKV基于内存映射MMAP,下面主要介绍内存映射相关内容:

1.1 定义

Linux通过将一个虚拟内存区域与一个磁盘上的对象关联起来,以初始化这个虚拟内存区域的内容,这个过程称为内存映射(memory mapping)

资深安卓研发大佬详解MMKV:谷歌都推荐的轻量级存储方案

1.2 读写原理

  1. 对文件进行mmap后,会在进程的虚拟内存分配地址空间 & 创建映射关系

  2. 实现映射关系后,就可以采用指针的方式读写操作这一段内存,而系统会自动回写到对应的文件磁盘上。

1.3 优势

  1. 减少数据拷贝次数:对文件的读写操作只需要从磁盘到用户主存的一次数据拷贝过程;

  2. 操作数据速度快:使用逻辑内存对磁盘文件进行映射,操作内存就相当于操作文件,不需要开启线程,和操作内存的速度一样快;MMAP提供一段可供随时写入的内存块,App 只管往里面写数据,由操作系统如内存不足、进程退出等时候负责将内存回写到文件,不必担心 crash 导致数据丢失

  3. 操作灵活、安全性高:通过 mmap 内存映射文件,提供了一段可供随时写入的内存块,App 只管往里面写数据,由操作系统负责将内存回写到文件,不必担心 crash 导致数据丢失。

2. 数据存储方式:Protobuf


MMKV的序列化/反序列化使用 protobuf 实现,其采用了以 T - V 方式对数据进行二进制数据流存储,空间占存少、数据量精简,能以最少的数据量能表示最多的信息。

3. 写入方式

最后

题外话,我在一线互联网企业工作十余年里,指导过不少同行后辈。帮助很多人得到了学习和成长。

我意识到有很多经验和知识值得分享给大家,也可以通过我们的能力和经验解答大家在IT学习中的很多困惑,所以在工作繁忙的情况下还是坚持各种整理和分享。但苦于知识传播途径有限,很多程序员朋友无法获得正确的资料得到学习提升,故此将并将重要的Android进阶资料包括自定义view、性能优化、MVC与MVP与MVVM三大框架的区别、NDK技术、阿里面试题精编汇总、常见源码分析等学习资料。

【Android思维脑图(技能树)】

知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。

Android开发8年,阿里、百度一面惨被吊打!我是否应该转行了?

【Android进阶学习视频】、【全套Android面试秘籍】

希望我能够用我的力量帮助更多迷茫、困惑的朋友们,帮助大家在IT道路上学习和发展
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
、常见源码分析等学习资料。

【Android思维脑图(技能树)】

知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。

[外链图片转存中…(img-rcE7xN2l-1715397070280)]

【Android进阶学习视频】、【全套Android面试秘籍】

希望我能够用我的力量帮助更多迷茫、困惑的朋友们,帮助大家在IT道路上学习和发展
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 5
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值