我创建了实时数据,它发出单个事件,如 example 中所示.
接下来是我的问题: 如何在 LiveData 中的值发生变化时仅通知最后订阅的观察者?
我想到的是将观察者存储在 SingleLiveData 类的链表中,然后仅当传递的观察者与列表的最后一个元素相同时才调用 super.observe
。
我不确定这是否是最佳方法。
我想使用此机制将 FAB 单击事件从 Activity 传播到显示在 ViewPager 内部的 fragment 。 fragment 是动态添加到 View 分页器适配器的,所以假设我们知道 fragment 的顺序。
最佳答案
最后,我找到了解决此问题的方法。我不得不放弃发出单个事件的实时数据,因为它无法按照我需要的方式运行。
取而代之的是,我使用了简单的可变实时数据,它发出一个事件对象,该对象包装了一个数据,如 article 的最后一段所示。作者:何塞·阿尔塞雷卡。
我在 View 寻呼机中显示 fragment ,所以我当时只有一个可见 fragment 。
所以我的 View 模型看起来像这样:
class ActionViewModel : ViewModel() {
private val onCreateLiveData: MutableLiveData<Event<String>> = MutableLiveData()
fun observeOnCreateEvent(): LiveData<Event<String>> = onCreateLiveData
fun onCreateCollectionClick(message: String) {
this.onCreateLiveData.value = Event(message)
}
}
事件包装类实现如下所示:
/*Used as a wrapper for data that is exposed via a LiveData that represents an
event.*/
open class Event<out T>(private val content: T) {
var hasBeenHandled = false
private set // Allow external read but not write
/**
* Returns the content and prevents its use again.
*/
fun getContentIfNotHandled(): T? {
return if (hasBeenHandled) {
null
} else {
hasBeenHandled = true
content
}
}
/**
* Returns the content, even if it's already been handled.
*/
fun peekContent(): T = content
}
现在在 fragment 中我们可以观察到这样的事件:
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
actionViewModel = ViewModelProviders.of(requireActivity()).get(ActionViewModel::class.java)
actionViewModel.observeOnCreateEvent()
.observe(this, Observer {
it?.takeIf { userVisibleHint }?.getContentIfNotHandled()?.let {
//DO what ever is needed
}
})
}
fragment userVisibleHint 如果 fragment 当前对用户可见,属性将返回 true。因为我们当时只展示一个 fragment ,所以这对我们有用。这意味着 fragment 将仅访问可见的事件数据。
此外,事件包装器的实现只允许读取一次值,因此每次 Observer 获取此事件时,它的值将为 null,我们将忽略它。
结论:通过这种方式,我们模拟了仅通知最后订阅的观察者的单个事件实时数据。
关于android - 如何创建发出单个事件并仅通知最后订阅的观察者的 LiveData?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51757164/