0

我有一个ViewModel显示transactionstransactionSummaryLiveData的句柄

// Transactions
val transactions: LiveData<List<Transaction>> = getTransactionByDateUseCase
    .getTransactionsByDate(dateRange.startDate, dateRange.endDate)

// Transaction Summary
private var _transactionsSummary = MutableLiveData<TransactionSummary>()
val transactionsSummary: LiveData<TransactionSummary> = _transactionsSummary

transactions来自Room的返回LiveData<T>类似,transactionSummary但有一个转换过程来获取事务的摘要,如incomeexpense

下面是改造过程:

private fun getTransactionsSummary(dateRange: DateRange): LiveData<TransactionSummary> {
    return Transformations.map(
        getTransactionByDateUseCase.getTransactionsByDate(dateRange.startDate, dateRange.endDate)
    ) { data ->
        getTransactionSummaryUseCase.transactions = data
        val summary = getTransactionSummaryUseCase.getSummaries(dateRange)
        summary
    }
}

但是,当我从 Fragment 请求更新时,transactionSummary总是返回NULL

getFetchTransactionViewModel().updateTransactionSummary(selectedDateRange)

这是我里面的更新功能ViewModel

fun updateTransactionSummary(dateRange: DateRange) {
    _transactionsSummary.value = getTransactionsSummary(dateRange).value
}

我观察到transactionSummary这样的:

    getFetchTransactionViewModel().transactionsSummary.observe(this, Observer {
            textViewIncome.text = it.totalIncome.formatCurrency(isVisibleCurrency = true)
            textViewExpense.text = it.totalExpense.formatCurrency(isVisibleCurrency = true)
            textViewBalance.text = (it.totalIncome - it.totalExpense).formatCurrency(isVisibleCurrency = true)
    })

是否有任何遗漏的步骤或不正确的实现?

谢谢

4

2 回答 2

0

我自己也遇到了完全相同的问题。似乎使用 Transformations.map 创建的 LiveData 对象(通常可能还有 MediatorLiveData )仅在第一个观察者订阅它们(活动状态)后才初始化它们的值。在此之前,即使源 LiveData 有值,它也会返回值 null(非活动状态)。这发生在你的函数updateTransactionSummary()中。

解决方案 1:您可以直接使用转换后的 LiveData,而不是尝试从中提取值并将其放入另一个 MutableLiveData。像这样的东西:

private var _transactionsSummary = getTransactionsSummary(dateRange)
val transactionsSummary: LiveData<TransactionSummary> = _transactionsSummary

然后你就不需要这个updateTransactionSummary()功能了。

此外,您可以直接使用事务 LiveData 作为转换的源,而不是获取第二个:

private fun getTransactionsSummary(dateRange: DateRange): LiveData<TransactionSummary> {
    return Transformations.map(transactions) { data ->
        getTransactionSummaryUseCase.transactions = data
        val summary = getTransactionSummaryUseCase.getSummaries(dateRange)
        summary
    }
}

解决方案 2:在您的评论中,您说您想通过另一个事件触发摘要更新。在这种情况下,使用 LiveData 转换似乎是多余的:

private fun getTransactionsSummary(dateRange: DateRange): TransactionSummary {
   getTransactionSummaryUseCase.transactions = transaction.value
   return getTransactionSummaryUseCase.getSummaries(dateRange)
}

fun updateTransactionSummary(dateRange: DateRange) {
    _transactionsSummary.value = getTransactionsSummary(dateRange)
}
于 2020-06-27T17:01:27.643 回答
0
fun updateTransactionSummary(dateRange: DateRange) {
    _transactionsSummary.value = getTransactionsSummary(dateRange).value
}

这不是使用 LiveData 的正确方法。

DateRange也是一个可变参数,必须作为输入传播到异步数据加载。

这意味着它必须包含在一个MutableLiveData<DateRange>, 然后.switchMap'd 到正确的查询结果中。

// Transaction Summary
//private var _transactionsSummary = MutableLiveData<TransactionSummary>()

private val dateRange = MutableLiveData<DateRange>()

val transactionsSummary: LiveData<TransactionSummary> = dateRange.switchMap { date ->
    getTransactionsSummary(dateRange)
}

很有可能DateRange在进程死亡时不应该忘记,在这种情况下,如果DateRange实际上是Parcelableor Serializable,那么它应该使用SavedStateHandlea自动持久化到 Bundle ViewModelprivate val dateRange = savedStateHandle.getLiveData<DateRange>("dateRange")如果可能的话使用。

于 2020-06-27T22:54:57.997 回答