Android之SharedPreferences(SP)
SharedPreferences 是 Android 提供的一种轻量级的数据存储方式,重要用于存储简朴的键值对(key-value pairs)。它非常得当保存应用的设置信息、用户设置、应用状态等小型数据。在 Android 中,SharedPreferences 默认情况下是进程内的,即每个进程都有自己独立的 SharedPreferences 实例。
为了在多进程中可靠地共享数据,保举使用 ContentProvider。ContentProvider 是 Android 提供的用于进程间通信的机制,可以确保数据的同等性和同步性。
MMKV——基于 mmap 的高性能通用 key-value 组件
概念
[*]键值对存储:SharedPreferences 使用键值对(key-value)的方式存储数据,键是字符串,值可以是多种根本数据类型。
[*]持久化存储:存储在 SharedPreferences 中的数据会被持久化到设备的文件体系中,纵然应用被关闭或设备重启,数据仍旧存在。
[*]轻量级:得当存储少量的简朴数据,不得当存储大量数据或复杂对象。
使用
1.获取 SharedPreferences 实例
可以通过以下几种方式获取 SharedPreferences 实例:
1.1 通过 Context 获取默认的 SharedPreferences 文件
SharedPreferences sharedPreferences = getSharedPreferences("MyPrefs", Context.MODE_PRIVATE);
1.2 通过 Activity 获取默认的 SharedPreferences 文件
SharedPreferences sharedPreferences = getPreferences(Context.MODE_PRIVATE);
1.3 通过 PreferenceManager 获取默认的 SharedPreferences 文件
SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context);
2.存储数据
使用 SharedPreferences.Editor 对象来存储数据。Editor 提供了多种 put 方法来存储不同类型的数据。
SharedPreferences sharedPreferences = getSharedPreferences("MyPrefs", Context.MODE_PRIVATE);
SharedPreferences.Editor editor = sharedPreferences.edit();editor.putString("username", "JohnDoe");editor.putInt("age", 30);editor.putBoolean("isLoggedIn", true);// 提交更改editor.apply(); // 大概使用 commit() 方法
[*]apply():异步提交,不会壅闭主线程,保举使用(没有返回值)。
[*]commit():同步提交,返回一个布尔值表示提交是否成功(有返回值)。
3.读取数据
SharedPreferences sharedPreferences = getSharedPreferences("MyPrefs", Context.MODE_PRIVATE);
String username = sharedPreferences.getString("username", null);int age = sharedPreferences.getInt("age", 0);boolean isLoggedIn = sharedPreferences.getBoolean("isLoggedIn", false); 4.删除数据
可以通过 Editor 的 remove 方法删除特定键的数据,大概使用 clear 方法删除全部数据。
SharedPreferences sharedPreferences = getSharedPreferences("MyPrefs", Context.MODE_PRIVATE);
SharedPreferences.Editor editor = sharedPreferences.edit();// 删除特定键的数据editor.remove("username");// 删除全部数据editor.clear();// 提交更改editor.apply(); // 大概使用 commit() 方法 5.监听数据变革
可以通过 SharedPreferences.OnSharedPreferenceChangeListener 接口监听数据的变革。
SharedPreferences sharedPreferences = getSharedPreferences("MyPrefs", Context.MODE_PRIVATE);
SharedPreferences.OnSharedPreferenceChangeListener listener = new SharedPreferences.OnSharedPreferenceChangeListener() { @Override public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key) { // 当数据发生变革时的处理逻辑 if (key.equals("username")) { String newUsername = sharedPreferences.getString(key, null); // 处理新的用户名 } }};// 注册监听器sharedPreferences.registerOnSharedPreferenceChangeListener(listener);// 取消注册监听器sharedPreferences.unregisterOnSharedPreferenceChangeListener(listener); 使用场景
[*]用户设置:保存用户的偏好设置,如主题、通知设置等。
[*]应用状态:保存应用的状态信息,如上次登录时间、用户会话信息等。
[*]简朴数据缓存:保存一些简朴的临时数据,如搜索汗青、用户输入等。
注意事项
[*]实用范围:SharedPreferences 得当存储少量的简朴数据,不得当存储大量数据或复杂对象。
[*]线程安全:SharedPreferences 是线程安全的,可以在多个线程中安全地使用。
[*]性能:使用 apply() 方法举行异步提交,避免壅闭主线程。
SharedPreferences 为什么是线程安全的
SharedPreferences线程安全
SharedPreferences 的线程安全性重要体现在以下几个方面:
1. 数据存储和读取的线程安全性
[*]读取操作:SharedPreferences 的读取操作是线程安全的。读取数据时不会有并发问题,因为读取操作是无锁的,并且不会修改底层数据布局。
[*]写入操作:SharedPreferences 的写入操作通过 SharedPreferences.Editor 类举行。Editor 类使用内部锁机制来确保写入操作的原子性。
2. 使用 apply() 方法举行异步提交
[*]apply() 方法:apply() 方法在后台线程中异步提交数据,不会壅闭主线程。它通过 SharedPreferencesImpl 类的 enqueueDiskWrite 方法将写操作参加队列,并在后台线程中实验。这种方式避免了多个线程同时写入数据时的冲突。
3. 使用 commit() 方法举行同步提交
[*]commit() 方法:commit() 方法是同步提交数据,返回一个布尔值表示提交是否成功。它使用内部锁机制确保在提交过程中不会有其他线程举行写操作。虽然 commit() 是线程安全的,但它会壅闭调用它的线程,因此不保举在主线程中使用。
4. 数据监听的线程安全性
[*]数据监听:SharedPreferences 提供了 registerOnSharedPreferenceChangeListener 方法来注册数据变革监听器。当数据发生变革时,全部注册的监听器会被通知。SharedPreferences 内部通过同步机制确保在通知监听器时的线程安全性。
内部实现
SharedPreferences 的线程安全性重要通过以下几个内部实现来包管:
[*]锁机制:在写入操作中使用内部锁机制(如 synchronized 关键字)来确保写操作的原子性和线程安全性。
[*]异步处理:使用 apply() 方法举行异步提交,将写操作放入队列并在后台线程中实验,避免壅闭主线程。
[*]内存缓存:SharedPreferences 使用内存缓存来存储数据,淘汰频繁的 I/O 操作。读取操作直接从内存缓存中获取数据,包管了高效性和线程安全性。
以下是 SharedPreferences 内部使用锁机制确保线程安全性的简化示例:
public final class SharedPreferencesImpl implements SharedPreferences {
private final Object mLock = new Object();
private final Map<String, Object> mMap;
@Override
public Map<String, ?> getAll() {
synchronized (mLock) {
return new HashMap<>(mMap);
}
}
@Override
public Editor edit() {
return new EditorImpl();
}
public final class EditorImpl implements Editor {
private final Map<String, Object> mModified;
@Override
public Editor putString(String key, String value) {
synchronized (mLock) {
mModified.put(key, value);
return this;
}
}
@Override
public void apply() {
final Runnable writeToDiskRunnable = new Runnable() {
@Override
public void run() {
synchronized (mLock) {
// 将 mModified 写入磁盘
}
}
};
// 异步执行写操作
new Thread(writeToDiskRunnable).start();
}
@Override
public boolean commit() {
synchronized (mLock) {
// 将 mModified 写入磁盘
return true;
}
}
}
}
在这个简化的示例中,mLock 用于同步访问共享资源(如 mMap 和 mModified),以确保线程安全性。
SharedPreferences 线程安全总结
SharedPreferences 的线程安全性是通过内部锁机制、异步处理和内存缓存等方式实现的。这些机制确保了在多线程环境下举行数据存储和读取操作时的安全性和高效性。
SharedPreferences 工具类示例
import android.content.Context;
import android.content.SharedPreferences;
public class SharedPrefsHelper {
private static final String PREF_NAME = "MyPrefs";
private static SharedPrefsHelper instance;
private SharedPreferences sharedPreferences;
private SharedPreferences.Editor editor;
private SharedPrefsHelper(Context context) {
sharedPreferences = context.getSharedPreferences(PREF_NAME, Context.MODE_PRIVATE);
editor = sharedPreferences.edit();
}
public static synchronized SharedPrefsHelper getInstance(Context context) {
if (instance == null) {
instance = new SharedPrefsHelper(context.getApplicationContext());
}
return instance;
}
// 存储字符串
public void putString(String key, String value) {
editor.putString(key, value);
editor.apply();
}
// 读取字符串
public String getString(String key, String defaultValue) {
return sharedPreferences.getString(key, defaultValue);
}
// 存储整数
public void putInt(String key, int value) {
editor.putInt(key, value);
editor.apply();
}
// 读取整数
public int getInt(String key, int defaultValue) {
return sharedPreferences.getInt(key, defaultValue);
}
// 存储布尔值
public void putBoolean(String key, boolean value) {
editor.putBoolean(key, value);
editor.apply();
}
// 读取布尔值
public boolean getBoolean(String key, boolean defaultValue) {
return sharedPreferences.getBoolean(key, defaultValue);
}
// 存储浮点数
public void putFloat(String key, float value) {
editor.putFloat(key, value);
editor.apply();
}
// 读取浮点数
public float getFloat(String key, float defaultValue) {
return sharedPreferences.getFloat(key, defaultValue);
}
// 存储长整型
public void putLong(String key, long value) {
editor.putLong(key, value);
editor.apply();
}
// 读取长整型
public long getLong(String key, long defaultValue) {
return sharedPreferences.getLong(key, defaultValue);
}
// 删除特定键的数据
public void remove(String key) {
editor.remove(key);
editor.apply();
}
// 清除所有数据
public void clear() {
editor.clear();
editor.apply();
}
}
在应用中使用 SharedPrefsHelper 工具类举行数据存储和读取:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 获取 SharedPrefsHelper 实例
SharedPrefsHelper sharedPrefsHelper = SharedPrefsHelper.getInstance(this);
// 存储数据
sharedPrefsHelper.putString("username", "Alan");
sharedPrefsHelper.putInt("age", 30);
sharedPrefsHelper.putBoolean("isLoggedIn", true);
// 读取数据
String username = sharedPrefsHelper.getString("username", "defaultName");
int age = sharedPrefsHelper.getInt("age", 0);
boolean isLoggedIn = sharedPrefsHelper.getBoolean("isLoggedIn", false);
// 显示读取的数据
Log.d("MainActivity", "Username: " + username);
Log.d("MainActivity", "Age: " + age);
Log.d("MainActivity", "Is Logged In: " + isLoggedIn);
}
}
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页:
[1]