引入
(测试操作机器是华为Mate 20 Pro 128G,Android 10,每组重复1k次,时间单位是ms)
可以看出MMKV的耗时比其他耗时少的离谱。再看多进程下的性能:
不必多说。再看和DataStore的对比:
简介
根据MMKV官方文档所言
MMKV------基于 mmap 的高性能通用 key-value 组件
MMKV 是基于 mmap 内存映射的 key-value 组件,底层序列化/反序列化使用 protobuf 实现,性能高,稳定性强。从 2015 年中至今在微信上使用,其性能和稳定性经过了时间的验证。近期也已移植到 Android / macOS / Win32 / POSIX 平台,一并开源。
MMKV 原理
- 内存准备
通过 mmap 内存映射文件,提供一段可供随时写入的内存块,App 只管往里面写数据,由操作系统负责将内存回写到文件,不必担心 crash 导致数据丢失。 - 数据组织
数据序列化方面我们选用 protobuf 协议,pb 在性能和空间占用上都有不错的表现。 - 写入优化
考虑到主要使用场景是频繁地进行写入更新,我们需要有增量更新的能力。我们考虑将增量 kv 对象序列化后,append 到内存末尾。 - 空间增长
使用 append 实现增量更新带来了一个新的问题,就是不断 append 的话,文件大小会增长得不可控。我们需要在性能和空间上做个折中。
在as上的添加依赖:
dependencies {
implementation 'com.tencent:mmkv:1.3.1'
// "1.3.1" 可用任何可用版本替代
}
使用
MMKV 的使用非常简单,所有变更立马生效,无需调用 sync
、apply
。
配置 MMKV 根目录
-
在 App 启动时初始化 MMKV,设定 MMKV 的根目录(files/mmkv/),例如在
Application
里:javapublic void onCreate() { super.onCreate(); String rootDir = MMKV.initialize(this); }
CRUD
基本存储
在CRUD这上面,和sp一样都是采取k-v键值对的操作存储:
kotlin
MMKV kv = MMKV.defaultMMKV(); //MMKV 提供一个全局的实例,可以直接使用
kv.encode("bool", true);
System.out.println("bool: " + kv.decodeBool("bool"));
kv.encode("int", Integer.MIN_VALUE);
System.out.println("int: " + kv.decodeInt("int"));
kv.encode("long", Long.MAX_VALUE);
System.out.println("long: " + kv.decodeLong("long"));
kv.encode("float", -3.14f);
System.out.println("float: " + kv.decodeFloat("float"));
kv.encode("double", Double.MIN_VALUE);
System.out.println("double: " + kv.decodeDouble("double"));
kv.encode("string", "Hello from mmkv");
System.out.println("string: " + kv.decodeString("string"));
byte[] bytes = {'m', 'm', 'k', 'v'};
kv.encode("bytes", bytes);
System.out.println("bytes: " + new String(kv.decodeBytes("bytes")));
存储的调用方法为
encode(k, v);
取出方法为
decode数据类型(k)
删除
java
MMKV kv = MMKV.defaultMMKV(); //所有操作需要这个实例
kv.removeValueForKey("key"); //移除括号内键值对应的数据
kv.removeValuesForKeys(new String[]{"int", "long"}); //移除多个数据
boolean hasBool = kv.containsKey("bool"); //查询某个键值对应的数据是否存在
区别存储
不同业务,不同存储,可以单独创建自己的实例
java
MMKV kv = MMKV.mmkvWithID("MyID");
kv.encode("key", "value");
多进程访问
需要多进程访问 ,在初始化的时候加上标志位 MMKV.MULTI_PROCESS_MODE
java
MMKV kv = MMKV.mmkvWithID("InterProcessKV", MMKV.MULTI_PROCESS_MODE);
kv.encode("key", value);
支持的数据类型
boolean
int
long
float
double
byte[]
String
Set<String>
以及任何实现了Parcelable
的类型(可以存对象!!)
进阶技巧就不说了,我用不太到,可参考MMKV官方进阶技巧