SharePreference 做为Android应用程序中经常用到的数据持久化一种手段,总共有两部分数据,分别是:
1 内存级缓存
2 文件缓存 xml形式存储,内部key-value结构
内存级缓存主要是为了提高数据的读取速度,文件缓存才是真正的数据存储
1 数据的存储过程: 按照写文件时同步存储和异步存储 分为 同步commit() 和 异步apply()两种,
具体的流程:
1)首先写数据到 内存级缓存,这部分调用 commit()和apply()没有区别,内部调用SharePreferenceImpl.java
MemoryCommitResult mcr = commitToMemory(); // 写数据到缓存
2) 写数据到文件,
commit()操作 会先写数据到内存缓存,然后创建一个Runnable,通过调用下面的语句将Runnable加入到写文件的队列中
QueuedWork.queue(writeToDiskRunnable, false);// 执行writeToDiskRunnable时不能延迟
apply()操作 会先写数据到内存缓存,然后创建一个Runnable,通过调用下面的语句将Runnable加入到写文件的队列中
QueuedWork.queue(writeToDiskRunnable, true);// 延迟执行Runnable
QueueWork.java
/**
* Queue a work-runnable for processing asynchronously.
*
* @param work The new runnable to process
* @param shouldDelay If the message should be delayed
*/
public static void queue(Runnable work, boolean shouldDelay) {
Handler handler = getHandler(); //此handler绑定了一个HandlerThread,运行于工作线程
synchronized (sLock) {
sWork.add(work);
if (shouldDelay && sCanDelay) {
handler.sendEmptyMessageDelayed(QueuedWorkHandler.MSG_RUN, DELAY);// 100ms以后执行
} else {
handler.sendEmptyMessage(QueuedWorkHandler.MSG_RUN);// 立即执行
}
}
}
SharePreference中所有的内存级别缓存都是线程安全的,内部通过使用Synchronized 维护
备注: SharePreference 是线程安全的,内部的commit,apply都实现了线程同步,但是不支持多进程访问,除了初始化时从xml文件读取数据,其他时候都是内存级别的,所以数据不能共享
xml 时可以实现数据共享的,但是多个进程同时访问同一个xml文件,是无法保证数据的同步,正确性
SharePreference写数据到缓存时,只会将有变化的数据重新写入文件, 另外,则使用backup文件(初始化时使用,如果写文件的过程中出错,则使用backup恢复数据)
针对多进程共享SharePreference,可以通过ContentProvider包括一层,能不用SharePreference实现
commit 是在当前线程中写数据到文件
apply在异步线程中写数据到文件
android 26之前 ,apply导致anr问题
http://www.cloudchou.com/android/post-988.html
解决办法:是不用apply直接自己异步commit. 也有人用反射把那个QueuedWork的sFinishers变量弄成empty