Android(安卓)SharePreference 总结
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
更多相关文章
- 一款常用的 Squid 日志分析工具
- GitHub 标星 8K+!一款开源替代 ls 的工具你值得拥有!
- “罗永浩抖音首秀”销售数据的可视化大屏是怎么做出来的呢?
- Nginx系列教程(三)| 一文带你读懂Nginx的负载均衡
- RHEL 6 下 DHCP+TFTP+FTP+PXE+Kickstart 实现无人值守安装
- Linux 环境下实战 Rsync 备份工具及配置 rsync+inotify 实时同步
- 不吹不黑!GitHub 上帮助人们学习编码的 12 个资源,错过血亏...
- Android实现系统级屏幕录制(下)
- android小入门