“刷新”房间数据库提供的 LiveData 的最佳方式

JJ *_*obs 7 android-lifecycle android-fragments kotlin android-room kotlin-coroutines

我目前正在使用 Room 存储单词翻译列表,并将查询作为 LiveData 返回以监视插入和更新。然而,当我需要重新获取不同源语言的翻译时,我遇到了问题(我当前的策略是将实时数据重新分配给房间查询的结果)。

我使用以下 SQL 查询获取要翻译的语言以及特定语言的翻译

@Dao
interface TranslationDatabaseDao {
   ...

   //Returns all pairs of languages to translate to/from
   @Query("SELECT * FROM language_pairs")
   fun getAllLanguagePairs(): LiveData<List<LanguagePair>>

   //Returns translations with the specified source language
   @Query("SELECT * FROM translations WHERE sourceLanguage = :language")
   fun getTranslations(language: String): LiveData<List<TranslationResult>>

   ...
}
Run Code Online (Sandbox Code Playgroud)

我在实例化视图模型时调用它,并在用户更改语言时再次调用它(见changeLanguage(...)下文)。

class translationViewmodel(private val database: TranslationDatabaseDao, initLanguage: String): ViewModel() {
   ...

   val languages: LiveData<List<LanguagePair>> = database.getAllLanguagePairs()
   val currentLanguages = Transformations.map(languages) { allLanguages ->
      allLanguages?.let {
         it[0] //Get the first language in the list
      }
   }

   var translations: LiveData<List<TranslationResult>> = database.getTranslations(initLanguage)

   ...

   fun changeLanguage(language: String) {
      coroutinesScope.launch {
         translations = withContext(Dispatchers.IO) {
            database.getTranslations(language)
         }
      }
   }

   ...
}
Run Code Online (Sandbox Code Playgroud)

重新分配翻译列表会导致观察实时数据的片段继续观察旧的实时数据,因此我的策略是在更改语言时删除观察者并在片段中重新分配它。这似乎不是最好的解决方案,而且我也不确定如何确保在删除前一个观察者并重新创建它之前已加载数据(我正在使用协程,所以我需要一种回调方法当实时数据返回时)。

可能的解决方案

  1. 让数据库仅返回一个 List 对象,而不是 livedata。然后我可以将其存储在可变的实时数据中。这并不理想,因为插入和更新不会自动反映在返回的列表中,因此每次更改特定语言的翻译列表时,我都必须重新获取数据。
  2. 获取所有翻译,然后在每次语言发生变化时过滤它们。一旦语言列表变大,这将占用大量资源。

所以我的问题是:有没有更好的方法在语言更改后重新分配实时数据?或者更确切地说,是否有一种可接受的方式让片段知道这种重新分配?

Val*_*kov 6

LiveData您可以为该语言再创建一个,并使用switchMap将其映射到翻译,例如:

class translationViewmodel(private val database: TranslationDatabaseDao, initLanguage: String): ViewModel() {
  val language = MutableLiveData<String>("en")
  val translations = language.switchMap { language -> 
    database.getTranslations(language)
  }
  ...
}
Run Code Online (Sandbox Code Playgroud)

现在,只需更改语言,翻译也会更新

viewmodel.language.value = "es"
Run Code Online (Sandbox Code Playgroud)