查询Firebase中的嵌套元素

时间:2016-04-12 17:51:23

标签: ios swift firebase firebase-realtime-database

我正在尝试使用swift查询Firebase集合。

我的收藏是这样的:

{
  "networks" : {
    "-KF9zQYGA1r_JiFam8gd" : {
      "category" : "friends",
      "members" : {
        "facebook:10154023600052295" : true
      },
      "name" : "My friends",
      "owner" : "facebook:10154023600052295",
      "picture" : "https://my.img/img.png"
    },
    "-KF9zQYZX6p_ra34DTGh" : {
      "category" : "friends",
      "members" : {
        "tototata" : true
      },
      "name" : "My friends2",
      "owner" : "facebook:10154023600052295",
      "picture" : "https://my.img/img.png"
    }
  }

我的查询是这样的:

let networksRef = ref.childByAppendingPath("networks")
                     .queryOrderedByChild("members")
                     .queryEqualToValue(true, childKey: uid)

然后我使用FirebaseUI填充TableView。

问题是查询没有返回任何内容,我也尝试使用这种查询:

let networksRef = ref.childByAppendingPath("networks")
                     .queryOrderedByChild("members")
                     .queryStartingAtValue(true, childKey: uid)
                     .queryEndingAtValue(true, childKey: uid)

要清楚“uid”是嵌套在“members”节点中的节点名称,其值为true。 (如图所示)

由于要下载的数据量,我无法在没有过滤的情况下获取所有节点。

我坚持了一天,我疯了......: - )

如果有人可以提供帮助?

1 个答案:

答案 0 :(得分:3)

正确的语法是:

let networksRef = ref.childByAppendingPath("networks")
                     .queryOrderedByChild("members/\(uid)")
                     .queryEqualToValue(true)

但请注意,您需要为每个uid提供一个索引才能有效地工作:

{
  "rules": {
    "networks": {
      ".indexOn": ["facebook:10154023600052295", "tototata"]
    }
  }
}

如果没有这些索引,Firebase客户端将下载所有数据并在本地进行过滤。结果将是相同的,但它会消耗大量不必要的带宽。

对此的正确解决方案要求您更改数据结构。在Firebase(以及大多数其他NoSQL数据库)中,您通常最终会以应用程序想要访问它的方式对数据进行建模。由于您正在寻找用户所属的网络,因此您应该按用户存储网络:

"networks_per_uid": {
  "facebook:10154023600052295": {
    "-KF9zQYGA1r_JiFam8gd": true
  },
  "tototata": {
    "-KF9zQYZX6p_ra34DTGh": true
  },
}

将这个所谓的索引添加到数据结构中称为非规范化,并在this blog postdocumentation on data structuring以及关于NoSQL data modeling的精彩文章中介绍。