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

你可能感兴趣的:(android)