MMKV 是基于 mmap 内存映射的 key-value 组件,底层序列化 / 反序列化使用 protobuf 实现,性能高,稳定性强。
MMKV 原理
通过 mmap 内存映射文件,提供一段可供随时写入的内存块,App 只管往里面写数据,由操作系统负责将内存回写到文件,不必担心 crash 导致数据丢失。
性能对比
将 MMKV 和 SharedPreferences、SQLite 进行对比, 重复读写操作 1k 次。
- 单进程性能
MMKV 在写入性能上远远超越 SharedPreferences & SQLite,在读取性能上也有相近或超越的表现。 - 多进程性能
MMKV 无论是在写入性能还是在读取性能,都远远超越 MultiProcessSharedPreferences & SQLite, MMKV 在 Android 多进程 key-value 存储组件上是不二之选。
引入
dependencies {implementation 'com.tencent:mmkv:1.2.14'
}
MMKV 的使用非常简单,所有变更立马生效,无需调用 sync
、apply
。 在 App 启动时初始化 MMKV,设定 MMKV 的根目录(files/mmkv/),例如在 MainActivity 里:
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);String rootDir = MMKV.initialize(this);System.out.println("mmkv root: " + rootDir);//……
}
MMKV 提供一个全局的实例,可以直接使用:
import com.tencent.mmkv.MMKV;
//……MMKV kv = MMKV.defaultMMKV();kv.encode("bool", true);
boolean bValue = kv.decodeBool("bool");kv.encode("int", Integer.MIN_VALUE);
int iValue = kv.decodeInt("int");kv.encode("string", "Hello from mmkv");
String str = kv.decodeString("string");
多进程
MMKV mmkv = MMKV.mmkvWithID("InterProcessKV", MMKV.MULTI_PROCESS_MODE);mmkv.encode("bool", true);
支持数据类型
- 支持以下 Java 语言基础类型:
boolean、int、long、float、double、byte[]
- 支持以下 Java 类和容器:
String、Set<String>
sharedprefrence迁移
- MMKV 提供了
importFromSharedPreferences()
函数,可以比较方便地迁移数据过来。 - MMKV 还额外实现了一遍
SharedPreferences
、SharedPreferences.Editor
这两个 interface,在迁移的时候只需两三行代码即可,其他 CRUD 操作代码都不用改。
private void testImportSharedPreferences() {
// SharedPreferences preferences = getSharedPreferences("myData", MODE_PRIVATE);MMKV preferences = MMKV.mmkvWithID("myData");// 迁移旧数据{SharedPreferences old_man = getSharedPreferences("myData", MODE_PRIVATE);preferences.importFromSharedPreferences(old_man);old_man.edit().clear().commit();}// 跟以前用法一样SharedPreferences.Editor editor = preferences.edit();editor.putBoolean("bool", true);editor.putInt("int", Integer.MIN_VALUE);editor.putLong("long", Long.MAX_VALUE);editor.putFloat("float", -3.14f);editor.putString("string", "hello, imported");HashSet<String> set = new HashSet<String>();set.add("W"); set.add("e"); set.add("C"); set.add("h"); set.add("a"); set.add("t");editor.putStringSet("string-set", set);// 无需调用 commit()//editor.commit();}