避免在 Android 的 SharedPreferences 中使用默认值

Avoid default values in Androids SharedPreferences

SharedPreferences 提供以下函数来检索字符串:

String getString(String key, @Nullable String defValue);

存储 Int 时,默认值不可为空:

int getInt(String key, int defValue);

我现在想要存储和检索可为 null 的 Int?,例如:

var timerDuration: Int?
    get() = prefs.getIntOrNull(TIMER_DURATION)
    set(value) { prefs.edit { putIntOrNull(TIMER_DURATION, value) } }

方案一:当value为null时,移除key

private fun SharedPreferences.getIntOrNull(key: String) =
    if (prefs.contains(key)) {
        getInt(key, 12345)
    } else {
        null
    }

private fun SharedPreferences.Editor.putIntOrNull(key: String, value: Int?) =
    if (value == null) {
        remove(key)
    } else {
        putInt(key, value)
    }

...但我不确定我是否会因为共享首选项的异步特性而在多次快速访问时遇到问题?


解决方案 2:对所有内容使用字符串选项:

private fun SharedPreferences.getIntOrNull(key: String) =
    prefs.getString(key, null)?.toIntOrNull()

private fun SharedPreferences.Editor.putIntOrNull(key: String, value: Int?) =
    putString(key, value?.toString())

...但是 .toIntOrNull() 对于这样一个简单的任务感觉开销很大?


我为什么要这样做?

我希望 Kotlin Multiplattform 允许我向我现有的 Android-App 添加一个 iOS-Version。 我的目标是完全用 Kotlin 编写的“核心”模块,没有平台特定的依赖项。 “核心”然后只使用这个接口,它由 Android & iOS - Apps

实现
interface SettingsStorage {

    var timerDuration: Int?

    [...]
}

因为我不希望有重复的默认值逻辑,所以我很乐意在我的“核心”模块中处理它


还有其他(更好的)选择吗?我觉得我正在重新发明轮子...

当您将 Int 存储到 SharedPreferences 时,它在内部存储为字符串。因此,对于获取,只需获取一个字符串,如果它是空值 returns,或者尝试将其解析为一个 Int。对于设置,只需将您的 Int 转换为 String 并使用 putString 方法将其放入。

var timerDuration: Int?
    get() = try { prefs.getString(TIMER_DURATION, null)?.toInt() } catch (e: Exception) { null }
    set(value) { prefs.edit { putString(TIMER_DURATION, value.toString()) } }

你的任何一种方法都可以。 SharedPreferences 是线程安全的。如果您在单独的进程中将清单中的服务设置为 运行 并且它与您的应用程序的其余部分同时修改首选项,则选项 1 只会真正失败。这不是你会不小心做的事情。

你的第二种方法没有任何额外开销,因为在幕后,SharedPreferences 都是字符串,所以你的代码基本上做与 SharedPreferences.getInt 相同的事情,除了 returns 默认值当首选项不存在或无法解析为 Int 时。