6

我正在尝试使用 Kotlin 对架构组件的实时数据进行基本实现,如下所示:

class MarketFragment : LifecycleFragment(){
......
 override fun onActivityCreated(savedInstanceState: Bundle?) {
        super.onActivityCreated(savedInstanceState)
        viewModel=ViewModelProviders.of(this).get(MarketViewModel::class.java)
        viewModel.book?.observe(this, Observer { book-> //updateUI })
....

我的 ViewModel 类是这样创建的:

class MarketViewModel : ViewModel()
{
    var book: MutableLiveData<Book>? =MutableLiveData()
    var repository: Repository?= Repository()

    init {
        update("parameter")
    }

    fun update(s: String)
    {
         book=repository?.getBook(s)

    }

}

和我的存储库:

fun getBook(bookSymbol:String):MutableLiveData<Book>
{
    val book=MutableLiveData<Book>()
    ......

            call . enqueue (object : Callback<Book> {
        override fun onResponse(call: Call<Book>?, response: Response<Book>?) {

            book.value=response?.body()

        }
     .....
    })
            return book
}

}

所有这些都非常有效,并且 UI 已按应有的方式更新,但只是第一次。如果我尝试手动调用从 UI 操作更新 viewModel,改造调用仍按预期工作,但新数据不会发送到 Fragment 中的观察者:

//this doesn't work:
viewModel.update("string")
//This returns false:
viewModel.book.hasActiveObservers()

观察者在第一次触发后变为非活动状态是预期的行为吗?

4

1 回答 1

10

每次调用时,您都在创建一个新的 MutableLiveData 实例getBooks

因此,您的观察者不再观察权利LiveData

为了解决这个

  1. 您的 ViewModel 应该只保留一个(不可变的)LiveData 实例
  2. 该不可变 LiveData 实例可能是:

    • 一个 MediatorLiveData,它的源是存储库的 LiveData
    • 存储库的 LiveData 的转换

这意味着存储库方法getBooks仅在 ViewModel 初始化时调用一次,并且要么刷新自身,要么您必须在存储库上调用另一个方法来触发刷新。

于 2017-08-18T22:38:21.137 回答