Cloud Firestore - 如何从两个集合中获取关系数据?

Sha*_*san 8 java android firebase google-cloud-firestore

我想在 Cloud Firestore 中定义这两个集合。一些示例数据如下所示:

播放列表:

  • 名称:“播放列表1”
  • songCount: //实时计算

歌曲:

  • 标题:“歌曲1”
  • 播放列表 ID:“播放列表 1”

  • 标题:“歌曲2”

  • 播放列表 ID:“播放列表 1”

  • 书名:“song3”

  • 播放列表 ID:“播放列表 1”

此外,在我的 Android 应用程序中,我想显示所有播放列表和所有歌曲的单独列表。但是,在显示所有播放列表的列表时,我想根据“songCount”中的值显示每个播放列表的名称以及每个播放列表中包含的歌曲数。

请告知我需要在 Cloud Firestore 中进行何种查询才能实现此目标。是否有JOIN可用的功能,或者我是否必须将 Songs 集合放入循环中并将具有 playlistId 的那些计数为“playlist1”,然后对要在播放列表列表中显示的所有播放列表重复此操作?我想要一个更智能的解决方案,但在互联网上找不到。

任何帮助将是一个很大的帮助。谢谢你的时间。

Ale*_*amo 9

有加入功能吗?

不幸的是,JOINFirestore 中没有查询。Firestore 中的查询是浅层的:它们只从运行查询的集合中获取项目。无法在单个查询中从顶级集合和其他集合或子集合中获取文档。Firestore 不支持一次性跨不同集合进行查询。单个查询只能使用单个集合中的文档属性。因此,我能想到的最简单的解决方案是使用类似于以下内容的数据库结构:

Firestore-root
   |
   --- playlists (collection)
   |     |
   |     --- playListId (document) //The unique id of the play list
   |           |
   |           --- name: "playlist1"
   |
   --- songs (collection)
        |
        --- playListId (document) //The same as the above playListId
               |
               --- playListSongs
                     |
                     --- songId
                          |
                          --- title: "song1"
Run Code Online (Sandbox Code Playgroud)

为了显示所有播放列表,只需将侦听器附加到playlists引用并获取所有playlist对象。如果您想获取与特定播放列表相对应的所有歌曲,只需将侦听器附加到songs\playListId\playListSongs并获取所有song对象即可。

关于与播放列表对应的所有歌曲的数量,我建议您从这篇文章中查看我的答案,我在其中解释了您可以实现的目标。因此,根据我回答的最后一部分,您的 Firebase 实时数据库结构应如下所示:

Firebase-Realtime-Database-root
    |
    --- playlists
          |
          --- playListIdOne: numberOfSongs
          |
          --- playListIdTwo: numberOfSongs
Run Code Online (Sandbox Code Playgroud)

编辑:

我不能说我完全理解它,特别是因为第一个答案涉及 Firestore,第二个涉及 Firebase 实时数据库。

我给你这个解决方案是因为如果你想在每次添加或删除歌曲时使用 Cloud Firestore 计算和更新元素,Firestore 将向你收取每次写入/删除操作的费用。Firebase 实时数据库有另一个定价计划,因此您无需为此付费。请参阅Firestore 定价请再次阅读直到最后,我在这篇文章中的回答。

此外,我无法真正获得计算歌曲数量的程序。

您可以通过以下方式从 Firestore 获取歌曲数量并将其写入 Firebase 实时数据库:

rootRef.collection("songs").document(playListId).collection("playListSongs")
.get().addOnCompleteListener(new OnCompleteListener<QuerySnapshot>() {
    @Override
    public void onComplete(@NonNull Task<QuerySnapshot> task) {
        if (task.isSuccessful()) {
            Log.d("TAG", task.getResult().size() + "");
            DatabaseReference rootRef = FirebaseDatabase.getInstance().getReference();
            DatabaseReference playListIdRef = rootRef.child("playlists").child(playListId);
            playListIdRef.setValue(task.getResult().size());
            
        } else {
            Log.d(TAG, "Error getting documents: ", task.getException());
        }
    }
});
Run Code Online (Sandbox Code Playgroud)

这就是你可以阅读的方式:

DatabaseReference rootRef = FirebaseDatabase.getInstance().getReference();
DatabaseReference playListIdRef = rootRef.child("playlists").child(playListId);
ValueEventListener valueEventListener = new ValueEventListener() {
    @Override
    public void onDataChange(DataSnapshot dataSnapshot) {
            long playListIdOne = dataSnapshot.getValue(String.Long);
            Log.d(TAG, "playListIdOne: " + playListIdOne);
    }

    @Override
    public void onCancelled(@NonNull DatabaseError databaseError) {
        Log.d(TAG, databaseError.getMessage());
    }
};
playListIdRef.addListenerForSingleValueEvent(valueEventListener);
Run Code Online (Sandbox Code Playgroud)

其次,根据您建议的结构,我如何获得所有播放列表中所有歌曲的列表?

在这种情况下,您应该创建另一个名为 的集合allSongs。数据库中的附加集合应如下所示:

Firestore-root
   |
   --- allSongs
         |
         --- songId
              |
              --- //song details
Run Code Online (Sandbox Code Playgroud)

这种做法被称为denormalizationFirebase 的常见做法。为了更好地理解,我建议您观看此视频,Firebase 数据库的非规范化是正常的。它适用于 Firebase 实时数据库,但相同的规则适用于 Cloud Firestore。

此外,当您复制数据时,您需要记住一件事。与添加数据的方式相同,您需要维护它。换句话说,如果你想更新/删除一个项目,你需要在它存在的每个地方都做。

请记住,同一首歌曲可以在多个播放列表中找到。

没关系,因为每首歌都有不同的 id。