全部 android asp.net C/C++ cshap IOS Java javascript nodejs perl php python ruby web容器 其他 前端 数据库 第三方平台 混合式APP 网络 系统 默认分类

Android数据存储之SharedPreferences浅析

0 61

  SharedPreferences 是 Android 数据存储方式中的一种,特别适合用来存储少量的、格式简单的数据,比如应用程序的各种配置信息,如是否打开音效,是否开启震动等等。

 SharedPreferences 存储数据的位置和格式

  SharedPreferences 将数据以键值对的形式,存储在 /data/data/<package name>/shared_prefs 目录下面,以 XML 的格式保存,该 XML 文件的根元素是 <map.../>,该元素里每个子元素代表一个 key-value 对。

 获取 SharedPreferences 的三种方式

  要想使用 SharedPreferences,首先需要得到 SharedPreferences 对象。SharedPreferences 本身是一个接口,因此程序无法直接创建 SharedPreferences 的对象,Android 提供了三种获取 SharedPreferences 对象的方法。

  1、Context 类的 getSharedPreferences(String name, int mode) 方法

      第一个参数用于指定 SharedPreferences 的文件名,第二个参数用于指定操作模式,传入 MODE_PRIVATE 或者 0 即可,表示只有当前应用程序可以操作SharedPreferences           文件。

  2、Activity 类的 getPreferences(int mode) 方法

     该方法的只接受一个参数,用于指定操作模式,和上面的方法一样。SharedPreferences 文件名以当前 Activity 的类名命名。该文件属于当前Activity私有的,只有当前Activity             可以操作。

  3、PreferenceManager 类的 getDefaultSharedPreferences(Context context) 方法

      这是一个静态方法,接受一个 Context 参数,以当前应用程序的报名作为前缀来命名 SharedPreferences 文件,整个应用程序可以操作。

 SharedPreferences 访问数据  

    SharedPreferences 中的方法

    得到 SharedPreferences 对象之后,则可以对数据进行读的操作。SharedPreferences 中提供的公共方法中的 getXXX 方法,即为读数据,比如

  getInt(String key, int defValue) 就是读取一个 int 型数据,传入第一个参数为存储数据的 key ,第二个参数为默认值,当 key 不存在的时候,会返回该默认值。

    getAll() 方法,会返回 SharedPreferences 中所有的键值对。

    contains(String key) 用来判断 SharedPreferences 所有键值对中是否包含键为 key 的键值对。

    另外 SharedPreferences 中还有一个内部接口,SharedPreferences.OnSharedPreferenceChangeListener ,用来声明一个回调,当某个 SharedPreferences 改变时会回调该方      法。SharedPreferences 提供了 registerOnSharedPreferenceChangeListener(SharedPreferences.OnSharedPreferenceChangeListener listener) 和                        unregisterOnSharedPreferenceChangeListener(OnSharedPreferenceChangeListener listener)  分别来注册和取消注册该回调。

    SharedPreferences.Editor 中的方法

    SharedPreferences 中还有一个方法 edit(),返回 SharedPreferences 的另一个内部接口 SharedPreferences.Editor 对象,该接口提供了诸如 putXXX 方法,很容易猜到是写        入数据的。同样拿 putInt(String key, int value) 来说,即往 SharedPreferences 文件中写入一个键值对。值得一提的是,调用 putXXX 方法后,还必须调用 commit() 或者apply()      方法,才能真正写入数据,这点很容易忘记(本人学习Android时曾踩过坑)。

    clear() 方法,清空 SharedPreferences 中的所有键值对。

    remove(String key) 方法,移除键为 key 的键值对,调用此方法后,需调用 commit 方法才能生效。

    最后还有两个方法, void apply() 和 boolean commit(),它们都是真正提交某次修改的。这里主要说说这两个方法的区别:

      • 从我上面写出的两个方法可以看出,apply 方法没有返回值,apply方法不会提示任何失败的提示。 commit 方法有返回值,表明修改是否提交成功。
      • 从用法上来说,调用 putXXX 方法之后,调用 apply 或者 commit 方法都可以提交,而调用 remove(String key) 方法之后,需调用 commit 方法提交。
      • 从本质上来说,apply是将修改数据原子提交到内存, 而后异步真正提交到硬件磁盘, 而commit是同步的提交到硬件磁盘,因此,在多个并发的提交commit的时候,他们会等待正在处理的commit保存到磁盘后在操作,从而降低了效率。而apply只是原子的提交到内容,后面有调用apply的函数的将会直接覆盖前面的内存数据,这样从一定程度上提高了很多效率。

    由于在一个进程中,sharedPreference是单实例,一般不会出现并发冲突,如果对提交的结果不关心的话,建议使用apply,当然需要确保提交成功且有后续操作的话,还是需要用        commit的。下面写一个简单例子,布局文具如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:orientation="vertical"
    tools:context="com.example.joy.sptest.MainActivity">

    <TextView
        android:id="@+id/tv_show"
        android:layout_marginTop="20dp"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"
        android:text="Hello World!" />

    <Button
        android:id="@+id/btn_write"
        android:layout_marginTop="10dp"
        android:layout_gravity="center_horizontal"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:onClick="onClick"
        android:text="写入数据"/>

    <Button
        android:id="@+id/btn_read"
        android:layout_marginTop="10dp"
        android:layout_gravity="center_horizontal"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:onClick="onClick"
        android:text="读取数据"/>

    <Button
        android:id="@+id/btn_modify"
        android:layout_marginTop="10dp"
        android:layout_gravity="center_horizontal"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:onClick="onClick"
        android:text="修改数据"/>

</LinearLayout>

程序代码如下:

package com.example.joy.sptest;

import android.content.SharedPreferences;
import android.preference.PreferenceManager;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {

    private TextView tvShow;
    private SharedPreferences sp;
    private SharedPreferences.Editor editor;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        tvShow = (TextView) findViewById(R.id.tv_show);
    }

    public void onClick(View v) {
        sp = PreferenceManager.getDefaultSharedPreferences(MainActivity.this);
        switch (v.getId()) {
            case R.id.btn_write:
                editor = sp.edit();
                editor.putString("name", "zhangsan");
                editor.putInt("age", 18);
                if (editor.commit()) {
                    Toast.makeText(MainActivity.this, "提交成功!", Toast.LENGTH_SHORT).show();
                } else {
                    Toast.makeText(MainActivity.this, "提交失败!", Toast.LENGTH_SHORT).show();
                }
                break;
            case R.id.btn_read:
                String name = sp.getString("name", "小明");
                int age = sp.getInt("age", 0);
                tvShow.setText("学生姓名:" + name + "  学生年龄:" + age);
                break;
            case R.id.btn_modify:
                editor = sp.edit();
                editor.putString("name", "张三");
                if (editor.commit()) {
                    Toast.makeText(MainActivity.this, "提交成功!", Toast.LENGTH_SHORT).show();
                } else {
                    Toast.makeText(MainActivity.this, "提交失败!", Toast.LENGTH_SHORT).show();
                }
                break;
        }
    }
}

  一个TextView,一开始运行会显示默认的 “Hello World!”,下面有三个button,点击写入数据,调用SharedPreferences,写入数据,点击读取数据,会从 SharedPreferences中读取数据,并显示在 textview 上,点击修改数据,则再次写入相同的 key ,替换掉之前的 value。程序运行结果如下:

                

    程序运行textview显示默认的“Hello World!”,点击读取数据,因为找不到 key ,所以返回指定的默认值,点击写入数据,写入姓名"zhangsan",年龄“18”。点击修改数据,将姓名       改为“张三”。

    通过DDMS文件可以查看,

                

    在 data/data/com.example.joy.sptest/shared_prefs 目录下面生成了一个 XML 文件:com.example.joy.sptest_preferences.xml。导出该文件,打开可以看到内容如下:

                

    最后贴出我自己封装后常用的读写 SharedPreferences 的方法:

public static void putSharedPreferences(Context context, String key, Object value) {
        String type = value.getClass().getSimpleName();
        SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context);
        SharedPreferences.Editor editor = sharedPreferences.edit();
        if ("Integer".equals(type)) {
            editor.putInt(key, (Integer) value);
        } else if ("Boolean".equals(type)) {
            editor.putBoolean(key, (Boolean) value);
        } else if ("String".equals(type)) {
            editor.putString(key, (String) value);
        } else if ("Float".equals(type)) {
            editor.putFloat(key, (Float) value);
        } else if ("Long".equals(type)) {
            editor.putLong(key, (Long) value);
        }
        editor.commit();
    }

    public static Object getSharedPreferences(Context context, String key, Object defValue) {
        String type = defValue.getClass().getSimpleName();
        SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context);
        //defValue为为默认值,如果当前获取不到数据就返回它
        if ("Integer".equals(type)) {
            return sharedPreferences.getInt(key, (Integer) defValue);
        } else if ("Boolean".equals(type)) {
            return sharedPreferences.getBoolean(key, (Boolean) defValue);
        } else if ("String".equals(type)) {
            return sharedPreferences.getString(key, (String) defValue);
        } else if ("Float".equals(type)) {
            return sharedPreferences.getFloat(key, (Float) defValue);
        } else if ("Long".equals(type)) {
            return sharedPreferences.getLong(key, (Long) defValue);
        }
        return null;
    }

热忱回答0

  • 加入年费大会员(20每年)
  • 热门标签

    猜你喜欢

    1. Android N特性解析
    2. android项目包改为library
    3. Android中Activity处理返回结果的实现方式
    4. React-Native 开发(一) Android环境部署,Hello react-native
    5. Android开发6:Service的使用(简单音乐播放器的实现)
    6. 老人桌面===花花绿绿的桌面
    7. Android 增量更新
    8. [Android]在Dagger 2中Activities和Subcomponents的多绑定(翻译)
    9. Android之探究viewGroup自定义子属性参数的获取流程
    10. Android手机USB调试安全闲扯(315晚会免费充电桩事件)

    衣食父母

    最近热帖

    1. Linux基础测试--11道题 6
    2. Java中双向链表的代码实现 12
    3. stl_config.h基本宏 7
    4. Struts2框架的基本使用 8
    5. 【WPF】学习笔记(一)——做一个简单的电子签名板 17
    6. 迁移 SQL Server 到 Azure SQL 实战 21
    7. 日期时间插件flatpickr.js使用方法 22
    8. 老司机带你开飞机 一: mssql on linux 安装指导 21
    9. 【代码学习】MYSQL数据库的常见操作 21
    10. 数字图像处理笔记与体会(二)——打开和显示图像 20

    随机文章

    1. javascript面向对象分层思维
    2. 解决:惠普HP LaserJet Pro M126a MFP 驱动 安装失败,及其它同类打印机失败问题
    3. RPC原来就是socket——RPC框架到dubbo的服务动态注册,服务路由,负载均衡演化
    4. Nuget 异常引用记录
    5. CSS3 border-radius边框圆角