如何将 Android DataStore 与多用户或文件一起使用

Wir*_*ing 17 android koin android-jetpack coroutinescope android-jetpack-datastore

我想使用 DataStore 存储一些首选项。但问题是我的应用程序可以有多个用户,因此需要将这些首选项存储在单独的文件中。我得到了一个仅使用一个用户的工作示例,但我正在努力支持多个用户。

这是我的代码的示例:

class DataStorageRepository(private val context: Context, private val userRepository: UserRepository) {

    private object PreferencesKeys {
        val SETTING_ONE = intPreferencesKey("setting_one")
    }

    // retrieve datastore for currently logged in user. 
    private val Context.dataStore: DataStore<Preferences> by preferencesDataStore(name = userRepository.currentRegistration().name)

    val userPreferencesFlow: Flow<UserPreferences> = context.dataStore.data.map { preferences ->
        val settingOne = preferences[PreferencesKeys.SETTING_ONE] ?: 0

        UserPreferences(settingOne)
    }

    suspend fun storeSettingOne(settingOne: Int) {
        context.dataStore.edit { preferences ->
            preferences[PreferencesKeys.SETTING_ONE] = settingOne
        }
    }

    data class UserPreferences(val lastUsedToAccountTab: Int)
}
Run Code Online (Sandbox Code Playgroud)

我正在使用Koin并尝试卸载DataStorageRepository注销并在登录时重新创建它,但数据存储似乎一直保持活动状态,直到应用程序被终止并且出现以下崩溃:

java.lang.IllegalStateException: 同一个文件有多个活动的 DataStore: [...] 您应该将 DataStore 维护为单例,或者确认同一文件上没有两个活动的 DataStore(通过确认范围是取消)。

我还尝试在注销时使用 aCoroutineScope并杀死它,但是在登录时重新创建范围后,数据存储似乎没有重新创建。

DataStore 是否支持关闭连接或处理多个文件的方法?

Moh*_*had 15

将此行放入伴生对象 { } 中

private val Context.dataStore: DataStore<Preferences> by preferencesDataStore(name = "settingPrefs")
Run Code Online (Sandbox Code Playgroud)

我的代码

class SettingPrefs(private val context: Context) {

    companion object {
        private val Context.dataStore: DataStore<Preferences> by preferencesDataStore(name = "settingPrefs")
        private val soundKey = booleanPreferencesKey("sound")
        private val vibrateKey = booleanPreferencesKey("vibrate")
    }

    val getSound: Flow<Boolean>
        get() = context.dataStore.data.map {
            it[soundKey] ?: true
        }

    suspend fun setSound(value: Boolean) {
        context.dataStore.edit { it[soundKey] = value }
    }

    val getVibration: Flow<Boolean>
        get() = context.dataStore.data.map {
            it[vibrateKey] ?: true
        }

    suspend fun setVibration(value: Boolean) {
        context.dataStore.edit { it[vibrateKey] = value }
    }
}
Run Code Online (Sandbox Code Playgroud)


use*_*210 8

您可以为不同的用户使用不同的密钥或手动保持 DataStore 单例


对于例外情况:

java.lang.IllegalStateException: 同一个文件有多个活动的 DataStore: [...] 您应该将 DataStore 维护为单例,或者确认同一文件上没有两个活动的 DataStore(通过确认范围是取消)。

androidx.datastore:datastore-*:1.0.0-alpha07发行了。

将其放在kotlin 文件的顶层,这样就只有一个实例。

private val Context.dataStore by preferencesDataStore("settings")

class Xxx{

}
Run Code Online (Sandbox Code Playgroud)

https://developer.android.com/jetpack/androidx/releases/datastore#1.0.0-alpha07

Context.createDataStore 扩展函数已被删除并替换为 globalDataStore 属性委托。在 kotlin 文件的顶层调用一次 globalDataStore。例如:

val Context.myDataStore by dataStore(...) 
Run Code Online (Sandbox Code Playgroud)

将其放在 kotlin 文件的顶层,这样就只有一个实例。(I57215,b/173726702)


Wir*_*ing 6

在我发布这个问题的那一刻,我找到了这个问题的解决方案。为了解决我的问题,我需要结合之前的两个解决方案。因此,注销时我卸载它DataStorageRepository,登录时我再次重新加载它。我还需要创建一个CoroutineScope在注销时取消的。

我的模块

val loggedInModule = module {
    single { DataStorageRepository(get(), get()) }
}
Run Code Online (Sandbox Code Playgroud)

我创建了一个范围并将其传递给 DataStore

var loggedInScope: CoroutineScope = CoroutineScope(Dispatchers.Default)

private val Context.dataStore: DataStore<Preferences> by preferencesDataStore(name = userRepository.currentRegistration().name, scope = loggedInScope)
Run Code Online (Sandbox Code Playgroud)

登录时

loggedInScope = CoroutineScope(Dispatchers.Default)
loadKoinModules(loggedInModule)
Run Code Online (Sandbox Code Playgroud)

注销时

loggedInScope.cancel()
unloadKoinModules(loggedInModule)
Run Code Online (Sandbox Code Playgroud)


Ald*_*wan 5

只需将声明数据存储区从 DataStorageRepository 类中取出即可

private val Context.dataStore: DataStore<Preferences> by preferencesDataStore(name="settings")

class DataStorageRepository(context: Context) {

   private var appContext = context.applicationContext

   val mData: Flow<String?> = appContext.dataStore.data.map { preferences ->
          preferences[YOUR_KEY]
   }

   suspend fun insertData(value: String) {
        appContext.dataStore.edit { preferences ->
             preferences[YOUR_KEY] = authToken
        }
   }

   companion object {
        private val KEY = stringPreferencesKey("data")
   }
}
Run Code Online (Sandbox Code Playgroud)