首页 > 解决方案 > (android firebase)确定分页时到达最后一项

问题描述

假设我在 Firebase 实时数据库中存储了 100 个项目(实际上它比 100 多得多)。所以我决定使用分页,我使用 Query#addValueEventListener() 函数进行查询,每个请求只加载 20 个项目,最后一个查询项目的键传递给 Query#startAt() 以获取接下来的 20 个项目。到目前为止效果很好

但问题是,当我到达 Firebase 数据库中的最后一项(第 100 项)后,它会将数据库中的所有项目(100 项)归还给我,以供下一个请求。目前,我必须检查传递给 Query#startAt() 的密钥 id 和最后一个查询项目的密钥 id 是否相等,如果为真,那么我已经到达终点。老实说,我认为这不是一个很好的解决方案

所以这里的任何人都有更好的解决方案,请帮助我。提前致谢!

这里的代码

private var lastKey: String? = null
fun requestItems(startAtId: String?){
    val query = dbRef.orderByKey()
    if(!startAtId.isNullOrBlank()){*the problem here*
        query.startAt(startAtId)
    }
    query.limitToFirst(10)
    query.addValueEventListener(object : ValueEventListener{
        override fun onDataChange(dataSnapshot: DataSnapshot) {
            val lastKey = it.children.last().key
            if (lastKey.equals(startAtId)){
                //has reached the last item
                //ignore the result
            }else{
                //do something with the result
                lastKey = latestKey
            }
        }
        override fun onCancelled(dataSnapshot: DatabaseError) {

        }

    })

}

更新 我刚刚意识到我面临的问题是在 requestItems() 函数中初始化 Query 对象(第 1 行到第 5 行)。它总是从 db 返回整个项目。我将该初始化块更改为波纹管块。我不知道与他们有什么不同,但它起到了分页的作用。

val query = if(startAtId.isNullOrBlank()){
        dbRef.orderByKey()
    }else{
        dbRef.orderByKey()
                .startAt(startAtId)

    }.limitToFirst(10)

但是仍然没有迹象表明它到达了 db 中的最后一项。到达终点后 Firebase 总是返回最后一项,我认为这是因为 startAt(startAtId) 函数而发生的,并且没有像 startAtAfter(startAtId) 这样的函数

我在此处附加了json 文件,它不是 db 中的真实数据,但它们具有相同的结构。

标签: androidfirebasefirebase-realtime-database

解决方案


FirebaseQuery对象是不可变的。每当您调用orderBy,startAt和其他方法之一时,它都会返回一个新Query对象。因此,您需要在变量中捕获该对象:

val query = dbRef.orderByKey()
if(!startAtId.isNullOrBlank()){
    query = query.startAt(startAtId)
}
query = query.limitToFirst(10)
...

这也是您更新的构造起作用的原因:您已将所有查询构建放入一个链并捕获结果。但是虽然它有效,但我个人发现结果的可读性不如上面的结果。


推荐阅读