SharedPreferences 详解(多进程,存取数组解决方案)

摘要:
实现SharedPreferences存储的步骤如下:根据Context获取SharedPrefeences对象,并使用edit()方法获取Editor对象。总之,由于sharedPreference是进程中的单个实例,因此通常不会发生并发冲突。如果您不关心提交的结果,建议使用apply。当然,如果需要确保成功提交和后续操作,仍然需要使用提交。它执行请求的修改以替换SharedPreferences中的任何数据。当两个编辑器同时修改首选项时,最后一次提交成功。因为SharedPreferences是进程中的单个实例。
一、SharedPreferences基本概念
文件保存路径:/data/data/<包名>/shared_prefs目录下目录下生成了一个SP.xml文件
SharedPreferences对象本身只能获取数据而不支持存储和修改,存储修改是通过Editor对象实现。
实现SharedPreferences存储的步骤如下:
  1. 根据Context获取SharedPreferences对象

  2. 利用edit()方法获取Editor对象。

  3. 通过Editor对象存储key-value键值对数据。

  4. 通过apply()或者commit()方法提交数据。

二、SharedPreferences相关api
  • SharedPreferences.Editorapi:完成数据写入操作

SharedPreferences 详解(多进程,存取数组解决方案)第1张

  • SharedPreferencesapi:完成数据读取操作

SharedPreferences 详解(多进程,存取数组解决方案)第2张

说明:所以的getXXX()方法,都支持默认值,即如果没有找到与当前key值对应的value,则返回我们自己设置的默认值。
  • commit()方法与apply()方法的比较

相同点:
  1. 二者都是提交preference修改数据

  2. 二者都是原子过程。

区别:
  1. apply没有返回值而commit返回boolean表明修改是否提交成功

  2. apply是将修改数据原子提交到内存,而后异步真正提交到硬件磁盘;而commit是同步的提交到硬件磁盘,因此,在多个并发的提交commit的时候,他们会等待正在处理的commit保存到磁盘后在操作,从而降低了效率。而apply只是原子的提交到内容,后面有调用apply的函数的将会直接覆盖前面的内存数据,这样从一定程度上提高了很多效率。

  3. apply方法不会提示任何失败的提示。

综合上述,由于在一个进程中,sharedPreference是单实例,一般不会出现并发冲突,如果对提交的结果不关心的话,建议使用apply,当然需要确保提交成功且有后续操作的话,还是需要用commit的。
  • 对api的理解

1.commit介绍:publicabstractbooleancommit()
修改你的preferences,从Editor到SharePreferences。它执行所请求的修改,替代SharedPreferences中的任何数据,当2个editor同时修改preferences,最后一个commit成功。如果不关注返回值或在程序的main线程使用时,推荐使用apply().
2.apply介绍:publicabstractvoidapply()
区别:commit将同步的将数据写到preferences;apply立即更改内存中的SharedPreferences,但是开始异步提交到磁盘中。保存失败你也不会得到任何提示信息,如果在这个sharedPreferences有另外一个editor执行一个定期的commit,此时一个apply依旧未完成。commit将被阻塞,直到所有异步操作完成,以及自己的commit。由于SharedPreferences在进程中是单实例的。在忽悠返回值的前提下,取代任何实例的commit或apply都是安全的。

三、对数据集合的支持——可以直接写入一个Set<String>类型的集合,但是默认只支持HashSet<String>类型,其他类型会被自动转型
Set<String>是一个接口,一般使用子类来完成相关的功能。

如果你希望传入的Set是一个有序的(跟插入时的顺序一样),那么使用LinkedHashSet<String>类型,构造好一个LinkedHashSet<String>对象之后,调用SharedPreferences.EditorputStringSet(Stringkey,Set<String>values)写入。按照常规的思维,通过SharedPreferences的getStringSet(Stringkey)方法读取出来数据,然后强制转为LinkedHashLinked<String>,可是出现异常了,系统提示不能将HashSet<String>转为LinkedHashSet<String>类型。也就是说取出来的值已经不再是我们写入的类型了。所以就只能使用HashSet<String>默认的排序类型,这只能满足那些不关系顺序的情况。

四、SharedPreferences一次存入多个有序数据解决方案(String类型为例)——使用字符串拼接(StringBuilder)的方式

public voidtestSharedPreferences(Context context) {
        // 写数据操作
        final String regularExpression = "#";
        final String key = "key";
        final String[] strings = { "1111", "2222", "3333"};
        final String defaultValue = "5555";
        final String fileName = "filename";
        final StringBuilder stringBuilder = newStringBuilder();
        for (int i = 0; i < strings.length; i++) {
            stringBuilder.append(strings[i]);
            if (i != strings.length - 1) {
                stringBuilder.append(regularExpression);
            }
        }
        SharedPreferences preferencesWrite = context.getSharedPreferences(fileName, Context.MODE_WORLD_READABLE);
        SharedPreferences.Editor editor = preferencesWrite.edit();
        editor.putString(key, stringBuilder.toString());
        editor.apply();
  
       // 读数据操作
        SharedPreferences preferencesRead =context.getSharedPreferences(fileName, Context.MODE_WORLD_READABLE);
        final String resultStr = preferencesRead.getString(key, defaultValue); // 没有对应的key则返回“5555”
        final String[] resultArray = resultStr.split(regularExpression);  // 解析数据,字符串分割
        for(String str : resultArray) {
            System.out.println(str);
        }
}

五、SharedPreferences多进程支持(虽然支持,但是还是不靠谱的,大量同时读写操作也会存在问题)

public static intgetMode() {
    //MODE_MULTI_PROCESS is always on in apps targeting Gingerbread
    //(Android 2.3) and below, and off by default in later versions
    return Build.VERSION.SDK_INT > 8 ? 4 : 0;
}
   
public voidtest() {
    SharedPreferences settings = context.getSharedPreferences("fileName", getMode());
}

下面是getSharedPreferences方法的定义和解释:

publicabstractSharedPreferencesgetSharedPreferences(Stringname,intmode)

AddedinAPIlevel1

Retrieveandholdthecontentsofthepreferencesfile'name',returningaSharedPreferencesthroughwhichyoucanretrieveandmodifyitsvalues.OnlyoneinstanceoftheSharedPreferencesobjectisreturnedtoanycallersforthesamename,meaningtheywillseeeachother'seditsassoonastheyaremade.

免责声明:文章转载自《SharedPreferences 详解(多进程,存取数组解决方案)》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇UVA 690 PipelineScheduling 位运算+dfs+剪枝SpringMVC基础知识下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

MD5Helper辅助类

DES加密和解密 public classMD5Helper { ///DES加密 ///sKey public string MD5Encrypt(string pToEncrypt, stringsKey) { DESCryp...

IDEA使用switch传入String编译不通过

今天在使用IDEA的时候,用到switch分支语句,传入String参数的时候一直报错,下面是源码报错截图: 看错误提示并没有提到switch支持String类型,不过ava1.7之后就支持String类型才对呀,于是想到了会不会是JDK问题,但是JDK用的是1.8呀!网上搜索才发现,会不会是编译环境版本过低的原因呢?于是查看对IDEA的ProjectS...

C#中String跟string的“区别”

  string是c#中的类,String是.net Framework的类(在C# IDE中不会显示蓝色) C# string映射为.net Framework的String 如果用string,编译器会把它编译成String,所以如果直接用String就可以让编译器少做一点点工作。   如果使用C#,建议使用string,比较符合规范 string始终...

Java 的设计模式之一装饰者模式

刚开始接触装饰者的设计模式,感觉挺难理解的,不够后来花了一个晚上的时间,终于有头绪了 装饰者设计模式:如果想对已经存在的对象进行装饰,那么就定义一个类,在类中对已经有的对象进行功能的增强或添加另外的行为,这个类就叫装饰者类。被修饰的类叫被装饰者类,是已经存在有的功能。在装饰者类之间又可以互相装饰 特点:          1.装饰类通过构造方法来接收被装饰...

Ext Js简单Grid创建使用及AJAX处理

在使用Ext Js对于Grid组件使用必不可少的,对于它的掌握也是需要的。简单贴一些代码,看看Grid的创建使用,就不细讲每一步了,代码注释还可以,不明白的可以在评论中写一下,或发邮件给我,一定帮助解答,欢迎交流。1.简单Ext JsGrid的创建使用(创建Ext.grid.GridPanel需要store(proxy\reader)\colModel)...

swift-字典

swift字典 在swift中,key:key值一定是可hash的,一定是独一无二的,swift的基本数据类型(String,Int,Float)都是可哈希的,所以都可以作为key值。 value:没有要求 直接上代码了,注释给大家标的很明白 //创建字典并赋值 let dict = ["name":"xiaoyu","age":12] l...