【问题标题】:What is the right way to use Firebase Database ChildEventListener with ViewModel?将 Firebase 数据库 ChildEventListener 与 ViewModel 一起使用的正确方法是什么?
【发布时间】:2026-01-25 10:45:02
【问题描述】:

在 Android 操作系统中,由于各种配置更改(例如方向更改),活动会被销毁并重新创建。但这也会导致更多的成本 - 例如重新获取数据以显示 UI。因此,ViewModel 被开发用于在活动真正销毁之前保持数据存在 - 在配置更改期间保持安全。

我一直在我的应用程序中使用 Firebase 实时数据库,我非常熟悉 Firebase SDK 中的 ChildEventListener。由于使用ViewModel 有很多优点 - 我有兴趣在我的应用程序中使用它。我已经阅读了关于ViewModel 架构以及 Firebase 的各种资料、博客:但它们仅使用 ValueEvent 侦听器。我在应用程序中显示RecyclerViewChildEventListener 非常适合。

我想维护一个使用ChildEventListenerViewModel 架构的列表。我已经编写了代码来使用ViewModel 主动获取数据。

class DataViewModel : ViewModel() {

    private var listenerRemovePending = false
    private val listener = MyChildEventListener()
    private val handler = Handler(Looper.getMainLooper())
    private lateinit var dataListRef: DatabaseReference

    var data: MutableLiveData<ArrayList<DataModel>> = MutableLiveData()
    val dataList: ArrayList<DataModel> = arrayListOf()

    fun getData(userId: String): LiveData<ArrayList<DataModel>> {
        dataListRef =
            FirebaseDatabase.getInstance().getReference("/data/$userId/data")
        dataListRef.addChildEventListener(object : ChildEventListener {
            override fun onCancelled(databaseError: DatabaseError) {
                Log.e(
                    LOG_TAG,
                    "Can't listen to query :dataListRef",
                    databaseError.toException()
                )
            }

            override fun onChildMoved(snapshot: DataSnapshot, previousChildName: String?) {
                TODO("Not yet implemented")
            }

            override fun onChildChanged(snapshot: DataSnapshot, previousChildName: String?) {
                TODO("Not yet implemented")
            }

            override fun onChildAdded(snapshot: DataSnapshot, previousChildName: String?) {
                if (snapshot.exists()) {
                    val item: DataModel? =
                        snapshot.getValue(DataModel::class.java)
                    if (item != null) {
                        dataList.add(item)
                    }
                    data.value = dataList
                }
            }

            override fun onChildRemoved(snapshot: DataSnapshot) {
                TODO("Not yet implemented")
            }
        })
        return data
    }

    companion object {
        private const val LOG_TAG = "DataViewModel"
    }
}

我对它的结果很满意。但这是一个主要问题 - 如果没有观察者(活动未激活),我希望此 ChildEventListener 被分离。目前,此侦听器处于活动状态,直到 Activity 结束 - 此 ViewModel 将保持其侦听器连接,直到调用 Activity 的 onDestroy 方法。

所以我的问题是如何从这里分离处于非活动状态的侦听器?

【问题讨论】:

    标签: android firebase kotlin android-lifecycle android-viewmodel


    【解决方案1】:

    您应该在 ViewModel 类中覆盖 onCleared() 函数。但是如果你想使用 MVVM 模式,你的网络请求应该在 SomRequestRemoteDataSource 中。

    override fun onCleared() {
        super.onCleared()
        dataListRef.removeEventListener(yourChildListener)
    }
    

    【讨论】: