2

当使用 Livedata 作为对 Room 中表的 select* 查询的返回类型时,我观察它,如果我更新/插入/删除该表中的条目,我会得到触发器。但是,当我尝试使用 Kotlin Flow 时,我只得到了 2 个触发器。

第一个触发器给出一个空值,因为状态流的初始值为空。第二个触发器是 Room 表中的条目列表。

如果我对数据库执行插入/删除操作,我会收到来自 StateFlow 的触发器。但是,如果我更新条目,Stateflow 不会触发。

注意:更新操作在数据库上正常工作。我使用数据库检查器进行了检查。

数据类 & DAO

@Entity
data class CartItem (
    @PrimaryKey
    val itemId: Int,
    var itemQuantity: Int=1
)

@Dao
interface CartDao {

    @Query("SELECT * FROM CartItem")
    fun getAllItems(): Flow<List<CartItem>>

    @Update
    suspend fun changeQuantityInCart(cartItem:CartItem)

    @Insert
    suspend fun insert(item: CartItem)

    @Delete
    suspend fun delete(cartItem:CartItem)
}

视图模型

val cartItems: StateFlow<List<CartItem>?> =
        repo.fetchCartItems().stateIn(viewModelScope, SharingStarted.Lazily, null)

分段

viewLifecycleOwner.lifecycleScope.launchWhenStarted {
            viewModel.cartItems.collect {
              Log.e("Update","Update")
 }
4

1 回答 1

3

我的陷阱是我正在像这样更新对象:

 currentItem.itemQuantity = currentItem.itemQuantity + 1
 changeQuantity(currentItem)

(是最初从DAO 中的流接收currentItem的类对象。)CartItemgetAllItems

changeQuantity乐趣称为changeQuantityInCartDAO 中的乐趣。


这导致 StateFlow 中 CartItem 对象的引用在调用数据库上的更新之前使用新的 itemQuantity 值保存对象的更新值。

之后,在 DAO 中调用 Update fun 时,DB 条目会更新,Flow 值会发生变化,但将其放入 Stateflow 中时不会检测到任何变化。因此,状态流不会触发,因为它是状态流与实时数据的不同之处。

在 livedata 的情况下,无论新值是否相同,它都会触发。

因此,要解决此错误,请不要在调用 DB 更新操作之前更改 stateFlow 中对象的值,如下所示:

val updatedCartItem = cartItem.copy(itemQuantity = cartItem.itemQuantity + 1)
changeQuantity(updatedCartItem)
于 2021-05-18T08:39:55.087 回答