列表的LiveData通知观察者,而无需调用setValue或postValue

时间:2019-07-09 14:52:47

标签: kotlin android-livedata mutablelivedata

我的视图模型中有一个MutableLiveData

val liveData = MutableLiveData<ArrayList<*>>()‍‍‍‍‍

我将来自端点调用的结果添加到此LiveData,如下所示:

liveData.value?.addAll(myList)

据我所知,MutableLiveData不应通知它是Observer,除非您对它执行setValuepostValue,但此时我的代码是运行,Observer会收到更改通知。

这怎么可能?


更新:

我遇到了一个更加奇怪的行为,该测试通过了,但是列表一次被打印出来:[]

    @Test
    fun `strange live data behavior`() {
        val myLiveData = MutableLiveData<ArrayList<Int>>()
        val observer = mock() as Observer<ArrayList<Int>>
        myLiveData.observeForever(observer)
        myLiveData.observeForever { println(it) }
        myLiveData.value = ArrayList()
        verify(observer).onChanged(ArrayList())
        myLiveData.value?.addAll(listOf(1, 2, 3, 4))
        val result = ArrayList<Int>()
        result.add(1)
        result.add(2)
        result.add(3)
        result.add(4)
        verify(observer).onChanged(result)
    }

1 个答案:

答案 0 :(得分:2)

LiveData发送通知时,LiveData不会发送该项目的副本。取而代之的是,它只是传递对持有的同一实例的引用。

这意味着如果您像这样修改LiveData内部的数据:

myLiveData.value?.addAll(listOf(1, 2, 3, 4))

观察者先前收到的ArrayList对象也将被修改而不会被调用Observer.onChanged() ,仅仅是因为它们是同一对象。这就是为什么在LiveData或任何观察者/反应模式中使用可变对象通常不是一个好主意的原因。

要验证onChanged()实际上仅被调用一次,请在测试结束时添加以下行:

    @Test
    fun `strange live data behavior`() {
        val myLiveData = MutableLiveData<ArrayList<Int>>()
        val observer = mock() as Observer<ArrayList<Int>>
        myLiveData.observeForever(observer)
        myLiveData.observeForever { println(it) }
        myLiveData.value = ArrayList()
        verify(observer).onChanged(ArrayList())
        myLiveData.value?.addAll(listOf(1, 2, 3, 4))
        val result = ArrayList<Int>()
        result.add(1)
        result.add(2)
        result.add(3)
        result.add(4)
        verify(observer).onChanged(result)

        // Below should pass because onChanged is only called once.
        verify(observer, times(1)).onChanged(any()) 
    }