ROOM/Android 使用 Repository 和 ViewModel 对列表进行排序 |科特林
ROOM/Android Sorting a List with Repository and ViewModel | Kotlin
当前问题: 第一次添加时,不可见地添加了一个对象(Deck)。只有从上下文菜单中选择了排序方法后,它才会出现。但是,每次都必须重复此操作才能更新屏幕。
问题应该出在存储库中,因为 getAllDecks 引用了 allDecks。就好像 allDecks 没有更新数据或者没有意识到它的 .value 正在改变。因为 allDecks.postValue() 将从数据库中获取列表。但这显然不是 LiveData。因此,它只做一次性的事情。所以。我将如何使存储库从数据库中读取不断更新的内容发生
我正在尝试对存储在存储库中的列表进行排序。我的 ViewModel 有一个引用存储在存储库中的项目的列表。当用户按下上下文菜单项时会发生排序,这将更改存储库中的列表。然而,这个动作似乎并没有奏效。
我尝试使用调试工具,它告诉我正在调用 Repository 方法并且正在重新分配内容。我相信这应该有效,因为我的 ViewModel 正在引用存储库,如果列表发生更改,MainActivity 会自动更新。
MainActivity 上下文菜单打开并对 onClick 作出反应的是根据所选项目进行排序更改的文本。因此,我知道它正在被调用。我还知道,因为我的更新、删除和插入查询正在运行,所以 MainActivity 正在监听 ViewModel 的(Decks)更改列表。
出于某种原因,我一直在努力寻找与我的和 Kotlin 类似的任何 Whosebug 帖子。
最后,我想知道是否有人可以为我做错的事情指出正确的方向。
和
另外,你如何调试数据库查询的问题。正如我发现的,当调试器转换为查看 SQLite 查询时,整个调试器进入循环。
亲切的问候,
PandaPlaysAll
主要Activity(缩写)
globalViewModel.sortBy(Sort.ALPHA_ASC) //Set default sorting
//Listen for livedata changes in ViewModel. if there is, update recycler view
globalViewModel.allDecks.observe(this, Observer { deck ->
deck?.let { adapter.setDecks(deck) }
})
override fun onContextItemSelected(item: MenuItem): Boolean {
when (item.itemId) {
R.id.sort_by_alpha_asc -> { globalViewModel.sortBy(Sort.ALPHA_ASC) ; currentSort = Sort.ALPHA_ASC ; contextMenuText.setText(R.string.sort_by_alpha_asc) ; return true; }
R.id.sort_by_alpha_desc -> { globalViewModel.sortBy(Sort.ALPHA_DES) ; currentSort = Sort.ALPHA_DES ; contextMenuText.setText(R.string.sort_by_alpha_des) ; return true; }
R.id.sort_by_completed_hidden -> { globalViewModel.sortBy(Sort.NON_COM) ; currentSort = Sort.NON_COM ; contextMenuText.setText(R.string.sort_by_non_complete) ; return true; }
R.id.sort_by_due_date -> { globalViewModel.sortBy(Sort.DUE_DATE) ; currentSort = Sort.DUE_DATE ; contextMenuText.setText(R.string.sort_by_due_date) ; return true; }
else -> return super.onContextItemSelected(item)
}
}
查看模型
private val repository: DeckRepository
val allDecks: LiveData<List<Deck>>
init {
val decksDao = FlashCardDB.getDatabase(application, viewModelScope).DeckDAO()
repository = DeckRepository(deckDao = decksDao)
allDecks = repository.getAllDecks()
}
fun sortBy(sortMethod: Sort) = viewModelScope.launch(Dispatchers.IO) {
when (sortMethod) {
Sort.ALPHA_ASC -> repository.sortBy(Sort.ALPHA_ASC)
Sort.ALPHA_DES -> repository.sortBy(Sort.ALPHA_DES)
Sort.NON_COM -> repository.sortBy(Sort.NON_COM)
Sort.DUE_DATE -> repository.sortBy(Sort.DUE_DATE)
}
}
DeckRepository
private var allDecks = MutableLiveData<List<Deck>>() //instantiate object
fun getAllDecks(): LiveData<List<Deck>> = allDecks //Repository handles livedata transmission. ViewModel references the actual Data.
suspend fun sortBy(sortingMethod: Sort) {
when (sortingMethod) {
Sort.ALPHA_ASC -> allDecks.postValue(deckDao.getDecksSortedByAlphaAsc())
Sort.ALPHA_DES -> allDecks.postValue(deckDao.getDecksSortedByAlphaDesc())
Sort.NON_COM -> allDecks.postValue(deckDao.getDecksSortedByNonCompleted())
Sort.DUE_DATE -> allDecks.postValue(deckDao.getDecksSortedByDueDate())
}
}
suspend fun insert(deck: Deck) {
deckDao.insert(deck)
}
数据库
//Sorting
@Query("SELECT * from deck_table ORDER BY title ASC")
fun getDecksSortedByAlphaAsc(): List<Deck>
@Query("SELECT * from deck_table ORDER BY title DESC")
fun getDecksSortedByAlphaDesc(): List<Deck>
@Query("SELECT * from deck_table WHERE completed=1 ORDER BY title ASC")
fun getDecksSortedByNonCompleted(): List<Deck>
@Query("SELECT * from deck_table ORDER BY date ASC")
fun getDecksSortedByDueDate(): List<Deck>
//Modifying
@Insert(onConflict = OnConflictStrategy.ABORT)
suspend fun insert(deck: Deck)
您的 activity 没有实时显示更改,因为您的存储库重新分配了它的 LiveData。虽然您让 activity 观察单个 LiveData 的方法是正确的,但实际上您应该只更改它的值,而不是参考(如果有意义的话)。
这是一个例子:
存储库
private val allDecks = MutableLiveData<List<Deck>>()
fun getAllDecks(): LiveData<List<Deck>> = allDecks
fun sortBy(sortingMethod: Sort) {
when (sortingMethod) {
/* If you're handling your DB operations with coroutines, this function
* should be suspendable and you should set the value to allDecks
* with postValue
*/
Sort.ALPHA_ASC -> allDecks.value = deckDao.getDecksSortedByAlphaAsc()
Sort.ALPHA_DES -> allDecks.value = deckDao.getDecksSortedByAlphaDesc()
Sort.NON_COM -> allDecks.value = deckDao.getDecksSortedByNonCompleted()
Sort.DUE_DATE -> allDecks.value = deckDao.getDecksSortedByDueDate()
}
}
因此,您的 DAO 查询将不再是 return LiveData,而是列表本身:
DAO
@Query("SELECT * from deck_table ORDER BY title ASC")
fun getDecksSortedByAlphaAsc(): List<Deck>
@Query("SELECT * from deck_table ORDER BY title DESC")
fun getDecksSortedByAlphaDesc(): List<Deck>
@Query("SELECT * from deck_table WHERE completed=1 ORDER BY title ASC")
fun getDecksSortedByNonCompleted(): List<Deck>
@Query("SELECT * from deck_table ORDER BY date ASC")
fun getDecksSortedByDueDate(): List<Deck>
当前问题: 第一次添加时,不可见地添加了一个对象(Deck)。只有从上下文菜单中选择了排序方法后,它才会出现。但是,每次都必须重复此操作才能更新屏幕。
问题应该出在存储库中,因为 getAllDecks 引用了 allDecks。就好像 allDecks 没有更新数据或者没有意识到它的 .value 正在改变。因为 allDecks.postValue() 将从数据库中获取列表。但这显然不是 LiveData。因此,它只做一次性的事情。所以。我将如何使存储库从数据库中读取不断更新的内容发生
我正在尝试对存储在存储库中的列表进行排序。我的 ViewModel 有一个引用存储在存储库中的项目的列表。当用户按下上下文菜单项时会发生排序,这将更改存储库中的列表。然而,这个动作似乎并没有奏效。
我尝试使用调试工具,它告诉我正在调用 Repository 方法并且正在重新分配内容。我相信这应该有效,因为我的 ViewModel 正在引用存储库,如果列表发生更改,MainActivity 会自动更新。
MainActivity 上下文菜单打开并对 onClick 作出反应的是根据所选项目进行排序更改的文本。因此,我知道它正在被调用。我还知道,因为我的更新、删除和插入查询正在运行,所以 MainActivity 正在监听 ViewModel 的(Decks)更改列表。
出于某种原因,我一直在努力寻找与我的和 Kotlin 类似的任何 Whosebug 帖子。
最后,我想知道是否有人可以为我做错的事情指出正确的方向。 和 另外,你如何调试数据库查询的问题。正如我发现的,当调试器转换为查看 SQLite 查询时,整个调试器进入循环。
亲切的问候,
PandaPlaysAll
主要Activity(缩写)
globalViewModel.sortBy(Sort.ALPHA_ASC) //Set default sorting
//Listen for livedata changes in ViewModel. if there is, update recycler view
globalViewModel.allDecks.observe(this, Observer { deck ->
deck?.let { adapter.setDecks(deck) }
})
override fun onContextItemSelected(item: MenuItem): Boolean {
when (item.itemId) {
R.id.sort_by_alpha_asc -> { globalViewModel.sortBy(Sort.ALPHA_ASC) ; currentSort = Sort.ALPHA_ASC ; contextMenuText.setText(R.string.sort_by_alpha_asc) ; return true; }
R.id.sort_by_alpha_desc -> { globalViewModel.sortBy(Sort.ALPHA_DES) ; currentSort = Sort.ALPHA_DES ; contextMenuText.setText(R.string.sort_by_alpha_des) ; return true; }
R.id.sort_by_completed_hidden -> { globalViewModel.sortBy(Sort.NON_COM) ; currentSort = Sort.NON_COM ; contextMenuText.setText(R.string.sort_by_non_complete) ; return true; }
R.id.sort_by_due_date -> { globalViewModel.sortBy(Sort.DUE_DATE) ; currentSort = Sort.DUE_DATE ; contextMenuText.setText(R.string.sort_by_due_date) ; return true; }
else -> return super.onContextItemSelected(item)
}
}
查看模型
private val repository: DeckRepository
val allDecks: LiveData<List<Deck>>
init {
val decksDao = FlashCardDB.getDatabase(application, viewModelScope).DeckDAO()
repository = DeckRepository(deckDao = decksDao)
allDecks = repository.getAllDecks()
}
fun sortBy(sortMethod: Sort) = viewModelScope.launch(Dispatchers.IO) {
when (sortMethod) {
Sort.ALPHA_ASC -> repository.sortBy(Sort.ALPHA_ASC)
Sort.ALPHA_DES -> repository.sortBy(Sort.ALPHA_DES)
Sort.NON_COM -> repository.sortBy(Sort.NON_COM)
Sort.DUE_DATE -> repository.sortBy(Sort.DUE_DATE)
}
}
DeckRepository
private var allDecks = MutableLiveData<List<Deck>>() //instantiate object
fun getAllDecks(): LiveData<List<Deck>> = allDecks //Repository handles livedata transmission. ViewModel references the actual Data.
suspend fun sortBy(sortingMethod: Sort) {
when (sortingMethod) {
Sort.ALPHA_ASC -> allDecks.postValue(deckDao.getDecksSortedByAlphaAsc())
Sort.ALPHA_DES -> allDecks.postValue(deckDao.getDecksSortedByAlphaDesc())
Sort.NON_COM -> allDecks.postValue(deckDao.getDecksSortedByNonCompleted())
Sort.DUE_DATE -> allDecks.postValue(deckDao.getDecksSortedByDueDate())
}
}
suspend fun insert(deck: Deck) {
deckDao.insert(deck)
}
数据库
//Sorting
@Query("SELECT * from deck_table ORDER BY title ASC")
fun getDecksSortedByAlphaAsc(): List<Deck>
@Query("SELECT * from deck_table ORDER BY title DESC")
fun getDecksSortedByAlphaDesc(): List<Deck>
@Query("SELECT * from deck_table WHERE completed=1 ORDER BY title ASC")
fun getDecksSortedByNonCompleted(): List<Deck>
@Query("SELECT * from deck_table ORDER BY date ASC")
fun getDecksSortedByDueDate(): List<Deck>
//Modifying
@Insert(onConflict = OnConflictStrategy.ABORT)
suspend fun insert(deck: Deck)
您的 activity 没有实时显示更改,因为您的存储库重新分配了它的 LiveData。虽然您让 activity 观察单个 LiveData 的方法是正确的,但实际上您应该只更改它的值,而不是参考(如果有意义的话)。
这是一个例子:
存储库
private val allDecks = MutableLiveData<List<Deck>>()
fun getAllDecks(): LiveData<List<Deck>> = allDecks
fun sortBy(sortingMethod: Sort) {
when (sortingMethod) {
/* If you're handling your DB operations with coroutines, this function
* should be suspendable and you should set the value to allDecks
* with postValue
*/
Sort.ALPHA_ASC -> allDecks.value = deckDao.getDecksSortedByAlphaAsc()
Sort.ALPHA_DES -> allDecks.value = deckDao.getDecksSortedByAlphaDesc()
Sort.NON_COM -> allDecks.value = deckDao.getDecksSortedByNonCompleted()
Sort.DUE_DATE -> allDecks.value = deckDao.getDecksSortedByDueDate()
}
}
因此,您的 DAO 查询将不再是 return LiveData,而是列表本身:
DAO
@Query("SELECT * from deck_table ORDER BY title ASC")
fun getDecksSortedByAlphaAsc(): List<Deck>
@Query("SELECT * from deck_table ORDER BY title DESC")
fun getDecksSortedByAlphaDesc(): List<Deck>
@Query("SELECT * from deck_table WHERE completed=1 ORDER BY title ASC")
fun getDecksSortedByNonCompleted(): List<Deck>
@Query("SELECT * from deck_table ORDER BY date ASC")
fun getDecksSortedByDueDate(): List<Deck>