Итерировать по дочерним снимкам в Firebase
У меня есть ресурс Firebase, который содержит несколько объектов, и я хотел бы перебрать их с помощью Swift. То, что я ожидал работать, является следующим (согласно документации Firebase)
https://www.firebase.com/docs/ios-api/Classes/FDataSnapshot.html
var ref = Firebase(url:MY_FIREBASE_URL)
ref.observeSingleEventOfType(.Value, withBlock: { snapshot in
println(snapshot.childrenCount) // I got the expected number of items
for rest in snapshot.children { //ERROR: "NSEnumerator" does not have a member named "Generator"
println(rest.value)
}
})
Таким образом, кажется, что существует проблема с итерацией Swift по объекту NSEnumerator, возвращенному Firebase.
Помощь действительно приветствуется.
5 ответов
Если я правильно прочитал документацию, это то, что вы хотите:
var ref = Firebase(url: MY_FIREBASE_URL)
ref.observeSingleEvent(of: .value) { snapshot in
print(snapshot.childrenCount) // I got the expected number of items
for rest in snapshot.children.allObjects as! [FIRDataSnapshot] {
print(rest.value)
}
}
Лучший способ может быть:
var ref = Firebase(url: MY_FIREBASE_URL)
ref.observeSingleEvent(of: .value) { snapshot in
print(snapshot.childrenCount) // I got the expected number of items
let enumerator = snapshot.children
while let rest = enumerator.nextObject() as? FIRDataSnapshot {
print(rest.value)
}
}
Первый метод требует NSEnumerator
вернуть массив всех объектов, которые затем могут быть перечислены обычным способом. Второй метод получает объекты по одному из NSEnumerator
и, вероятно, более эффективный.
В любом случае перечисляемые объекты FIRDataSnapshot
объекты, так что вам нужны приведения, чтобы вы могли получить доступ к value
имущество.
С помощью for-in
цикл:
С момента написания оригинального ответа в Свифт за 1,2 дня, язык эволюционировал. Теперь можно использовать for in
цикл, который работает непосредственно с перечислителями наряду с case let
назначить тип:
var ref = Firebase(url: MY_FIREBASE_URL)
ref.observeSingleEvent(of: .value) { snapshot in
print(snapshot.childrenCount) // I got the expected number of items
for case let rest as FIRDataSnapshot in snapshot.children {
print(rest.value)
}
}
Я только что преобразовал приведенный выше ответ в Swift 3:
ref = FIRDatabase.database().reference()
ref.observeSingleEvent(of: .value, with: { snapshot in
print(snapshot.childrenCount) // I got the expected number of items
for rest in snapshot.children.allObjects as! [FIRDataSnapshot] {
print(rest.value)
}
})
Лучший способ может быть:
ref = FIRDatabase.database().reference()
ref.observeSingleEvent(of: .value, with: { snapshot in
print(snapshot.childrenCount) // I got the expected number of items
let enumerator = snapshot.children
while let rest = enumerator.nextObject() as? FIRDataSnapshot {
print(rest.value)
}
})
Это довольно читабельно и отлично работает:
var ref = Firebase(url:MY_FIREBASE_URL)
ref.childByAppendingPath("some-child").observeSingleEventOfType(
FEventType.Value, withBlock: { (snapshot) -> Void in
for child in snapshot.children {
let childSnapshot = snapshot.childSnapshotForPath(child.key)
let someValue = childSnapshot.value["key"] as! String
}
})
ref = FIRDatabase.database().reference().child("exampleUsernames")
ref.observeSingleEvent(of: .value, with: { snapshot in
for rest in snapshot.children.allObjects as! [FIRDataSnapshot] {
guard let restDict = rest.value as? [String: Any] else { continue }
let username = restDict["username"] as? String
}
})
Firebase 4.0.1
Database.database().reference().child("key").observe(.value) { snapshot in
if let datas = snapshot.children.allObjects as? [DataSnapshot] {
let results = datas.flatMap({
($0.value as! [String: Any])["xxx"]
}
print(results)
}
}
Если у вас есть несколько ключей / значений, и вы хотите return
array
с dictionary
элементы, объявить массив:
var yourArray = [[String: Any]]()
затем измените тело блока на это:
let children = snapshot.children
while let rest = children.nextObject() as? DataSnapshot, let value = rest.value {
self.yourArray.append(value as! [String: Any])
}