[英]How to do remove an entire document collection in angularfire firestore
[英]How to subscribe to an entire collection and a document within it simultaneously without duplicating the amount of reads?
我正在为学习目的编写一个带有 Flutter 和 Firestore 的跨平台待办事项应用程序。 目前,我有以下设计,我想知道是否有更好的选择。
该应用程序的主屏幕之一显示所有任务的列表。 它通过订阅相应的 Firestore 集合来执行此操作,为简单起见,我们将其称为/tasks
。
FirebaseFirestore.instance.collection("tasks").snapshots()
任务列表视图中的每个磁贴都可以单击。 单击一个磁贴会打开一个新屏幕(带有Navigator.push
),显示有关该特定任务的详细信息。
重要的是,这个屏幕也需要实时更新,所以仅仅从主屏幕传递(本地,不可变的)任务 object 是不够的。 相反,此屏幕会订阅与该任务对应的单个 Firestore 文档。
FirebaseFirestore.instance.collection("tasks").doc(taskId).snapshots()
这在逻辑上对我来说是有意义的:详细信息页面只需要知道那个特定的文档,所以它只订阅它以避免接收不必要的更新。
问题是因为在详细信息屏幕打开时,主屏幕的集合范围订阅仍然存在,如果文档/tasks/{taskId}
更新,两个侦听器都会触发。 根据这个、 这个和这个问题中的答案,这意味着我将对对该文档的任何一次更新的两次(重复)读取收费。
此外,每个任务都可以有子任务。 这在 Firestore 中反映为每个任务的tasks
子集合。 例如,嵌套任务的路径可能是: /tasks/abc123/tasks/efg875/tasks/aay789
。 通过对"tasks"
使用集合组查询,主页可以显示所有任务,而不管嵌套。 上述详情页面还通过监听子集合来显示任务的子任务。 这允许对子任务进行复杂的查询(过滤、排序等),但同样的缺点是每次更新子任务都会重复读取。
我想到的替代设计是:
仅在整个应用程序范围内订阅整个任务集(无论是平面集合还是集合组查询),并在客户端上进行任何和所有选择、过滤等。 例如,任务的详细信息页面每次都会使用相同的集合范围订阅和 select 集合之外的适当任务。 任务/子任务的任何过滤和排序都将在客户端完成。
打开详情页取消全馆订阅,返回主界面重新开启。 这意味着当详细信息页面打开时,只会收到对该特定任务的更新,而不会被复制为两次读取。
这些设计中的任何一个看起来是最好的吗? 我还缺少其他更好的选择吗?
在你的应用程序中创建一个TaskService
或类似的东西来处理监听FirebaseFirestore.instance.collection("tasks").snapshots()
调用,然后在你的应用程序中,订阅该服务的更新而不是 Firebase 本身(你可以创建两个Stream
对象,一个用于全局更新,一个用于特定更新)。
然后,您的 Firebase 集合中只有一个正在阅读。 一切都在应用程序端处理。
伪代码:
class TaskService {
final List<Task> _tasks = [];
final StreamController<List<Task>> _signalOnTasks = StreamController.broadcast();
final StreamController<Task> _signalOnTask = StreamController.broadcast();
get List<Task> allTasks => _tasks;
Stream<List<Task>> get onTasks => _signalOnTasks.stream;
Stream<List<Task>> get onTask => _signalOnTask.stream;
void init() {
FirebaseFirestore.instance.collection("tasks").snapshots().listen(_onData);
}
void _onData(snapshot) {
/// get/update our tasks (maybe check for duplicates or whatever)
_tasks.addAll(snapshot.documents);
/// dispatch our signal streams
_signalOnTasks.add(snapshot.documents);
for(final task in snapshot.documents) {
_signalOnTask.add(task);
}
}
}
您可以让TaskService
和InheritedWidget
在任何地方访问它(或使用提供程序包),将您的侦听器添加到您感兴趣的任何 stream。您只需要将您的侦听器签入onTask
,这是正确的任务在对它做任何事情之前。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.