Android SharedPreferences 数据丢失问题
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android SharedPreferences 数据丢失问题相关的知识,希望对你有一定的参考价值。
参考技术A 最近由于需求迭代, K-V 的存储方式加入了加解密流程, 然后上线后,发现依赖于SharedPreferences 进行缓存的页面,发生了一些不可思议的报错,仿佛SharedPreferences 没有put数据到xml中一样.一开始不太相信,后面分析了下,确实是这样.commit 是同步提交, apply是异步处理. 为啥有这两种区分呢,那是因为SharedPreferences 如果用commit 来存储数据,数据量针对大数据那种,很容易造成ANR,因为要数据落地才能够进行后续相应的操作的话.你可以想象一下. 而用了apply 这种异步的话, apply的原理是开多一份xml 来进行读写, 等数据真实落地后,再删之前旧的那份xml. 那问题来了,既然这个过程是异步的,有没有可能会造成数据丢失,或者数据不准确呢? 答案是,确实如此.SharedPreferences 文件的加载使用了异步线程,而且加载线程并没有设置优先级,如果这个时候读取数据就需要等待文件加载线程的结束。这就导致主线程等待低优先线程锁的问题,比如一个 100KB 的 SP 文件读取等待时间大约需要 50 ~ 100ms,并且建议大家提前用预加载启动过程用到的 SP 文件。而且重要的一点,无论是commit 还是apply ,他在读写的过程都是全量写入的.
所以其中的数据读取量,根据相应的流而议,文件越大,那么相应的风险也就越大啦.因此SharedPreferences 也只适合那种轻量级的数据流的读写.
首先看入参:
方法是线程安全的, 但是跨线程呢? 如果你用了MODE_MULTI_PROCESS 的话,那别的线程就可以更改你的数据啦.但是如果你不用这种方式的话, 那数据就不能给到别的线程去读写,因此,这个流程,大家可以脑补一下.
1.不用用SharedPreferences 保存跳转入口的缓存,而应该利用Intent 去传递相应数据到Activity 或者Fragment.
2.不要存储大数据, 包括一些加解密,因为加密操作会导致string 的长度变大, 如果都是加密存储,那么内容可想而知.
3.可以区分用户级别数据和应用级别数据来进行处理,如果用户数据较大, 可以考虑一些开源库如MMKV,如果较小,需要加密处理保证安全的话,针对部分字段进行加解密即可.
4.应用尽量不要过分依赖SharedPreferences来进行相应的业务逻辑处理操作.考虑一些设计模式来避免这个过程吧.
Android :数据存储方案学习笔记之 SharedPreferences
SharedPreferences
1、SharedPreferences概述
要想使用 SharedPreferences 来存储数据,首先需要获取到 SharedPreferences 对象。Android 提供了三种方法得到 SharedPreferences 对象:
-
1、
Context 类
中的getSharedPreferences()
方法此方法接收两个参数,第一个参数指定
SharedPreferences
文件的名称,第二个参数指定操作模式,目前只有
MODE_PRIVATE
一种模式,和直接传入 0 效果相同。其他几种模式已被废弃。 -
2、
Activity 类
中的getPreferences()
方法
此方法和上面的方法相似,但只接收一个操作模式参数,使用这个方法时会自动将当前活动的类名作为SharedPreferences
的文件名。 -
3、
PreferenceManager 类
中的getDefaultSharedPreferences()
方法
这是一个静态方法,它接收一个 Context 参数,并自动使用当前应用程序的包名作为前缀来命名SharedPreferences
文件。
得到了 SharedPreferences
对象之后,分为三步实现向 SharedPreferences
文件中存储数据:
- (1)调用
SharedPreferences
对象的edit
()方法来获取一个SharedPreferences.Editor
对象。 - (2)向
SharedPreferences.Editor
对象中添加数据,如添加一个布尔型数据使用putBoolean
方法,添加一个字符串使用 putString()方法,以此类推。
(3) 调用 apply()方法将添加的数据提交,完成数据存储。
当然,SharedPreference在提交数据时也可用 Editor 的 commit 方法,两者区别如下:
apply
() 没有返回值,而commit
() 返回boolean
表明修改是否提交成功apply
() 将修改提交到内存,然后再异步提交到磁盘上;而commit
() 是同步提交到磁盘上。 谷歌建议:若在UI线程中,使用apply() 减少UI线程的阻塞(写到磁盘上是耗时操作)引起的卡顿。
新建一个项目
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<Button
android:id="@+id/save_data"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="保存数据" />
</LinearLayout>
放一个按钮,将一些数据存储到SharedPreferences 文件当中。然后修改 Activity 中的代码
public class SharePreferencesActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_share_preferences);
Button save_data = (Button) findViewById(R.id.save_data);
save_data.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
// 1.指定文件名为 wonderful,并得到 SharedPreferences.Editor 对象
SharedPreferences.Editor editor = getSharedPreferences("wonderful",MODE_PRIVATE).edit();
// 2.添加数据
editor.putString("name","开心wonderful");
editor.putInt("age",20);
editor.putBoolean("married",false);
// 3.数据提交
editor.apply();
}
});
}
}
点击按钮后,这时数据已保存成功了,生成了一个 wonderful.xml
文件
SharedPreferences
文件是使用 XML 格式来对数据进行管理的
2、从 SharedPreferences 中读取数据
SharedPreferences 对象中提供了一系列的 get 方法用于对存储的数据进行读取
每种 get 方法都对应了 SharedPreferences. Editor 中的一种 put 方法
这些 get 方法接收两个参数,第一个参数是键,即传入存储数据时使用的键;第二个参数是默认值
即当传入的键找不到对应的值时,返回默认值
修改上面项目中的布局代码:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<Button
android:id="@+id/save_data"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="保存数据" />
<Button
android:id="@+id/restore_data"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="读取数据" />
<TextView
android:id="@+id/show_data"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
</LinearLayout>
增加了一个还原数据的按钮和 TextView,点击按钮来从 SharedPreferences 文件中读取数据并在 TextView 中显示读取的数据
修改 Activity 中的代码:
public class SharePreferencesActivity extends AppCompatActivity {
private Button save_data,restore_data;
private TextView textView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_share_preferences);
save_data = (Button) findViewById(R.id.save_data);
save_data.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
// 1.指定文件名为 wonderful,并得到 SharedPreferences.Editor 对象
SharedPreferences.Editor editor = getSharedPreferences("wonderful",MODE_PRIVATE).edit();
// 2.添加不同类型的数据
editor.putString("name","开心wonderful");
editor.putInt("age",20);
editor.putBoolean("married",false);
// 3.数据提交
editor.apply();
}
});
textView = (TextView) findViewById(R.id.show_data);
restore_data = (Button) findViewById(R.id.restore_data);
restore_data.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
// 获得 SharedPreferences 对象
SharedPreferences pref = getSharedPreferences("wonderful",MODE_PRIVATE);
// 获取相应的值
String name = pref.getString("name","");
int age = pref.getInt("age",0);
boolean married = pref.getBoolean("married",false);
// 将获取到的值显示
textView.setText("name is " + name + ",age is "+ age + ",married is "+ married);
}
});
}
}
3 、实现记住密码功能
会用上一章广播的强制下线的例子
修改项目前,先来简单封装下关于 SharedPreferences 的工具类,如下:
public class PrefUtils {
private static final String PREF_NAME = "config";
/**
* 读取布尔数据
* @param ctx 上下文
* @param key 键
* @param defaultValue 默认值
* @return
*/
public static boolean getBoolean(Context ctx, String key,
boolean defaultValue) {
SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
Context.MODE_PRIVATE);
return sp.getBoolean(key, defaultValue);
}
/**
* 添加布尔数据
* @param ctx 上下文
* @param key 键
* @param value 添加的数据
*/
public static void setBoolean(Context ctx, String key, boolean value) {
SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
Context.MODE_PRIVATE);
sp.edit().putBoolean(key, value).apply();
}
/**
* 读取字符串
* @param ctx
* @param key
* @param defaultValue
* @return
*/
public static String getString(Context ctx, String key, String defaultValue) {
SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
Context.MODE_PRIVATE);
return sp.getString(key, defaultValue);
}
/**
* 添加字符串
* @param ctx
* @param key
* @param value
*/
public static void setString(Context ctx, String key, String value) {
SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
Context.MODE_PRIVATE);
sp.edit().putString(key, value).apply();
}
/**
* 读取int类型数据
* @param ctx
* @param key
* @param defaultValue
* @return
*/
public static int getInt(Context ctx, String key, int defaultValue) {
SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
Context.MODE_PRIVATE);
return sp.getInt(key, defaultValue);
}
/**
* 添加int类型数据
* @param ctx
* @param key
* @param value
*/
public static void setInt(Context ctx, String key, int value){
SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
Context.MODE_PRIVATE);
sp.edit().putInt(key, value).apply();
}
/**
* 将数据全部清除掉
* @param ctx
*/
public static void clear(Context ctx){
SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
Context.MODE_PRIVATE);
sp.edit().clear().apply();
}
}
编辑下登录界面,修改 activity_login.xml
中的代码:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<!--***************** 账号 *********************-->
<LinearLayout
android:orientation="horizontal"
android:layout_width="match_parent"
android:layout_height="60dp">
<TextView
android:layout_width="90dp"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:gravity="center"
android:textSize="18sp"
android:text="账号:"/>
<EditText
android:id="@+id/et_account"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:layout_gravity="center_vertical"/>
</LinearLayout>
<!--***************** 密码 *********************-->
<LinearLayout
android:orientation="horizontal"
android:layout_width="match_parent"
android:layout_height="60dp">
<TextView
android:layout_width="90dp"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:gravity="center"
android:textSize="18sp"
android:text="密码:"/>
<EditText
android:id="@+id/et_password"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:layout_gravity="center_vertical"
android:inputType="textPassword"/>
</LinearLayout>
<!--***************** 是否记住密码 *********************-->
<LinearLayout
android:orientation="horizontal"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<CheckBox
android:id="@+id/cb_remember_pass"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="10dp"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="18sp"
android:layout_gravity="center_vertical"
android:text="记住密码"/>
</LinearLayout>
<Button
android:id="@+id/btn_login"
android:layout_width="match_parent"
android:layout_height="60dp"
android:layout_margin="10dp"
android:text="登录"/>
</LinearLayout>
添加了个 CheckBox 来勾选记住密码,接着修改 LoginActivity 的代码:
public class LoginActivity extends BaseActivity {
private EditText et_account, et_password;
private CheckBox cb_remember_pass;
private Button btn_login;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
et_account = (EditText) findViewById(R.id.et_account);
et_password = (EditText) findViewById(R.id.et_password);
cb_remember_pass = (CheckBox) findViewById(R.id.cb_remember_pass);
btn_login = (Button) findViewById(R.id.btn_login);
Boolean isRemember = PrefUtils.getBoolean(this,"remember_pass",false);
if (isRemember){
// 将账号和密码都设置到文本框中
String account = PrefUtils.getString(this,"account","");
String password = PrefUtils.getString(this,"password","");
et_account.setText(account);
et_password.setText(password);
cb_remember_pass.setChecked(true);
}
btn_login.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
String account = et_account.getText().toString();
String password = et_password.getText().toString();
// 若账号是 wonderful 且密码是 123456,就认为登录成功
if (account.equals("wonderful") && password.equals("123456")){
// 检查复选框是否被勾选
if (cb_remember_pass.isChecked()){
// 保存数据到SharePreference文件中
PrefUtils.setBoolean(LoginActivity.this,"remember_pass",true);
PrefUtils.setString(LoginActivity.this,"account",account);
PrefUtils.setString(LoginActivity.this,"password",password);
}else {
// 清除SharePreference文件中的数据
PrefUtils.clear(LoginActivity.this);
}
// 登录成功跳转到主界面
IntentUtils.myIntent(LoginActivity.this,ForceOfflineActivity.class);
finish();
}else {
ToastUtils.showShort("账号或密码无效!");
}
}
});
}
@Override
protected int initLayoutId() {
return R.layout.activity_login;
}
}
参考
以上是关于Android SharedPreferences 数据丢失问题的主要内容,如果未能解决你的问题,请参考以下文章
android开发之路11(用SharedPreferences存储数据)
Android 工具类 SharedPreferences 封装
Android - 具有可序列化对象的 SharedPreferences