如何统计Firestore中某个集合下的文档数量?

2024-01-06

我想得到收藏参考 https://firebase.google.com/docs/reference/android/com/google/firebase/firestore/CollectionReference计数存在于 Cloud Firestore 上,我尝试通过以下方式获取它:

FirebaseFirestore db = FirebaseFirestore.getInstance();
    final CollectionReference postsCollection = db.collection("Posts");

    final TaskCompletionSource<Integer> source = new TaskCompletionSource<>();
    new Thread(new Runnable() {
        @Override
        public void run() {
            int fromWhereToStart = postsCollection.get().getResult().size();
            source.setResult(fromWhereToStart);
        }
    }).start();

    Task<Integer> task = source.getTask();
    task.addOnCompleteListener(new OnCompleteListener<Integer>() {
        @Override
        public void onComplete(@NonNull Task<Integer> task) {
            Log.e("Z_fromWhereToStart", "= " + task.getResult());
        }
    });

但不幸的是,我得到:

java.lang.IllegalStateException: Task is not yet complete

是否有另一种方法来获取另一种方法的计数来修复IllegalStateException?


Edit:2022 年 10 月 20 日

从现在开始,计算集合中的文档或查询返回的文档实际上是可能的,而不需要保留计数器。所以你可以使用新的方法来计算文档count() https://firebase.google.com/products/extensions/firestore-counter方法其中:

返回一个查询,该查询对该查询的结果集中的文档进行计数。

这一新功能是在今年的 Firebase 峰会上宣布的。请记住,此功能不会读取实际文档。所以根据官方文档 https://medium.com/firebase-tips-tricks/how-to-count-the-number-of-documents-in-a-firestore-collection-3bd0c719978f:

对于诸如 count() 之类的聚合查询,您需要为每批最多 1000 个与查询匹配的索引条目读取一个文档付费。对于匹配 0 个索引条目的聚合查询,读取一个文档的最低费用为 1。

例如,匹配 0 到 1000 个索引条目的 count() 操作按一次文档读取计费。对于匹配 1500 个索引条目的 count() 操作,您需要支付 2 次文档读取费用。


Edit:2021 年 7 月 10 日

最近,Firebase 添加了一个新扩展,名为分布式计数器 https://firebase.google.com/products/extensions/firestore-counter:

使用此扩展可以向您的应用程序添加高度可扩展的计数器服务。这对于计算病毒式操作或任何高速操作(例如查看、点赞或分享)的应用程序来说是理想的选择。

使用此扩展,您还可以突破每秒一次写入操作的最大限制。

这里还有一篇您可能感兴趣的文章:

  • 如何计算 Firestore 集合中的文档数量? https://medium.com/firebase-tips-tricks/how-to-count-the-number-of-documents-in-a-firestore-collection-3bd0c719978f

因为没有getDocumentCount()方法就像我们在 Firebase 实时数据库中一样,getChildrenCount()方法,实际计算您下面的所有文档的数量Posts从您的 Cloud Firestore 收集,请使用以下代码:

db.collection("Posts").get().addOnCompleteListener(new OnCompleteListener<QuerySnapshot>() {
    @Override
    public void onComplete(@NonNull Task<QuerySnapshot> task) {
        if (task.isSuccessful()) {
            int count = 0;
            for (DocumentSnapshot document : task.getResult()) {
                count++;
            }
            Log.d("TAG", count + "");
        } else {
            Log.d(TAG, "Error getting documents: ", task.getException());
        }
    }
});

or

db.collection("Posts").get().addOnCompleteListener(new OnCompleteListener<QuerySnapshot>() {
    @Override
    public void onComplete(@NonNull Task<QuerySnapshot> task) {
        if (task.isSuccessful()) {
            Log.d("TAG", task.getResult().size() + "");
        } else {
            Log.d(TAG, "Error getting documents: ", task.getException());
        }
    }
});

上面的示例对于小型数据集来说足够有效,但如果数据集较大,则不起作用。但是,还有两种方法可以实现同样的目标。

一种方法是使用云功能 https://cloud.google.com/functions/更新一个counter每次您添加或删除文档时Posts收藏。该技术也适用于大数据集。但请注意,在这种情况下,文档的添加和删除只能以小于或等于每秒 1 个的速率发生,如Cloud Firestore 配额和限制 https://firebase.google.com/docs/firestore/quotas。这是一个需要阅读的文档,但它几乎可以立即显示当前计数。

如果您需要超出此限制,则需要实施distributed counters如中所解释的分布式计数器的官方文档 https://firebase.google.com/docs/firestore/solutions/counters.

作为个人提示,不要将此类计数器存储在 Cloud Firestore 中,因为每次增加或减少计数器都会花费您read or a write手术。将此计数器托管在Firebase Realtime数据库几乎不花钱.

第二种方法是,不使用云功能,而是在客户端使用事务,在添加或删除文档的同时更新计数器。这样,你的计数器也会很准确,因为它是同时更新的。但在这种情况下,最重要的是您需要确保在添加或删除文档的任何位置都包含此逻辑。在这种情况下,您也可以免费使用 Firebase Realtime 数据库。

总之,将第一个代码用于小型数据集,第二个代码使用云函数,因为这是写入时尽力而为,第三个代码使用我上面向您解释的最后一个选项。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

如何统计Firestore中某个集合下的文档数量? 的相关文章

随机推荐