Eine FIRDatabaseReference abrufen
Zum Lesen oder Schreiben von Daten aus der Datenbank benötigen Sie eine Instanz von FIRDatabaseReference:
Swift
var ref: DatabaseReference! ref = Database.database().reference()
Objective-C
@property (strong, nonatomic) FIRDatabaseReference *ref; self.ref = [[FIRDatabase database] reference];
Listen lesen und schreiben
Daten an eine Liste anhängen
Verwenden Sie die Methode childByAutoId, um Daten an eine Liste in Mehrnutzeranwendungen anzuhängen. Die Methode childByAutoId generiert jedes Mal einen eindeutigen Schlüssel, wenn der angegebenen Firebase-Referenz ein neues untergeordnetes Element hinzugefügt wird. Durch die Verwendung dieser automatisch generierten Schlüssel für jedes neue Element in der Liste können mehrere Clients gleichzeitig untergeordnete Elemente am selben Ort hinzufügen, ohne dass es zu Schreibkonflikten kommt. Der von childByAutoId generierte eindeutige Schlüssel basiert auf einem Zeitstempel, sodass Listenelemente automatisch chronologisch sortiert werden.
Sie können die Referenz auf die neuen Daten, die von der Methode childByAutoId zurückgegeben werden, verwenden, um den Wert des automatisch generierten Schlüssels des untergeordneten Elements abzurufen oder Daten für das untergeordnete Element festzulegen.
Wenn Sie getKey für eine childByAutoId-Referenz aufrufen, wird der automatisch generierte Schlüssel zurückgegeben.
Sie können diese automatisch generierten Schlüssel verwenden, um die Struktur Ihrer Daten zu vereinfachen. Weitere Informationen finden Sie im Beispiel für die Datenverteilung .
Auf untergeordnete Ereignisse warten
Untergeordnete Ereignisse werden als Reaktion auf bestimmte Vorgänge ausgelöst, die für die untergeordneten Elemente eines Knotens ausgeführt werden. Dazu gehören beispielsweise das Hinzufügen eines neuen untergeordneten Elements über die Methode childByAutoId oder das Aktualisieren eines untergeordneten Elements über die Methode updateChildValues.
| Ereignistyp | Typische Verwendung |
|---|---|
FIRDataEventTypeChildAdded |
Listen von Elementen abrufen oder auf Ergänzungen einer Liste von Elementen warten. Dieses Ereignis wird einmal für jedes vorhandene untergeordnete Element und dann wieder jedes Mal ausgelöst, wenn dem angegebenen Pfad ein neues untergeordnetes Element hinzugefügt wird. Dem Listener wird ein Snapshot mit den Daten des neuen untergeordneten Elements übergeben. |
FIRDataEventTypeChildChanged |
Auf Änderungen an den Elementen in einer Liste warten. Dieses Ereignis wird jedes Mal ausgelöst, wenn ein untergeordneter Knoten geändert wird. Dazu gehören auch alle Änderungen an den Nachfolgern des untergeordneten Knotens. Der an den Ereignis-Listener übergebene Snapshot enthält die aktualisierten Daten für das untergeordnete Element. |
FIRDataEventTypeChildRemoved |
Auf Elemente warten, die aus einer Liste entfernt werden. Dieses Ereignis wird ausgelöst, wenn ein direkt untergeordnetes Element entfernt wird.Der an den Callback-Block übergebene Snapshot enthält die Daten für das entfernte untergeordnete Element. |
FIRDataEventTypeChildMoved |
Auf Änderungen an der Reihenfolge der Elemente in einer sortierten Liste warten.
Dieses Ereignis wird immer ausgelöst, wenn eine Aktualisierung zu einer Neusortierung des
untergeordneten Elements führt. Es wird mit Daten verwendet, die nach queryOrderedByChild
oder queryOrderedByValue sortiert sind.
|
Jedes dieser Ereignisse kann nützlich sein, um auf Änderungen an einem bestimmten Knoten in einer Datenbank zu warten. Beispielsweise kann eine Social-Blogging-App diese Methoden zusammen verwenden, um Aktivitäten in den Kommentaren eines Beitrags zu beobachten, wie unten gezeigt:
Swift
// Listen for new comments in the Firebase database commentsRef.observe(.childAdded, with: { (snapshot) -> Void in self.comments.append(snapshot) self.tableView.insertRows( at: [IndexPath(row: self.comments.count - 1, section: self.kSectionComments)], with: UITableView.RowAnimation.automatic ) }) // Listen for deleted comments in the Firebase database commentsRef.observe(.childRemoved, with: { (snapshot) -> Void in let index = self.indexOfMessage(snapshot) self.comments.remove(at: index) self.tableView.deleteRows( at: [IndexPath(row: index, section: self.kSectionComments)], with: UITableView.RowAnimation.automatic ) })
Objective-C
// Listen for new comments in the Firebase database [_commentsRef observeEventType:FIRDataEventTypeChildAdded withBlock:^(FIRDataSnapshot *snapshot) { [self.comments addObject:snapshot]; [self.tableView insertRowsAtIndexPaths:@[ [NSIndexPath indexPathForRow:self.comments.count - 1 inSection:kSectionComments] ] withRowAnimation:UITableViewRowAnimationAutomatic]; }]; // Listen for deleted comments in the Firebase database [_commentsRef observeEventType:FIRDataEventTypeChildRemoved withBlock:^(FIRDataSnapshot *snapshot) { int index = [self indexOfMessage:snapshot]; [self.comments removeObjectAtIndex:index]; [self.tableView deleteRowsAtIndexPaths:@[[NSIndexPath indexPathForRow:index inSection:kSectionComments]] withRowAnimation:UITableViewRowAnimationAutomatic]; }];
Auf Wert-Ereignisse warten
Das Warten auf untergeordnete Ereignisse ist zwar die empfohlene Methode zum Lesen von Datenlisten, es gibt aber auch Situationen, in denen das Warten auf Wert-Ereignisse für eine Listenreferenz nützlich ist.
Wenn Sie einen FIRDataEventTypeValue-Observer an eine Liste von Daten anhängen, wird die gesamte Liste der Daten als einzelner DataSnapshot zurückgegeben. Sie können dann eine Schleife durchlaufen, um auf einzelne untergeordnete Elemente zuzugreifen.
Auch wenn es nur eine Übereinstimmung für die Abfrage gibt, ist der Snapshot eine Liste, die nur ein Element enthält. Um auf das Element zuzugreifen, müssen Sie das Ergebnis durchlaufen:
Swift
_commentsRef.observe(.value) { snapshot in for child in snapshot.children { ... } }
Objective-C
[_commentsRef observeEventType:FIRDataEventTypeValue withBlock:^(FIRDataSnapshot *snapshot) { // Loop over children NSEnumerator *children = [snapshot children]; FIRDataSnapshot *child; while (child = [children nextObject]) { // ... } }];
Dieses Muster kann nützlich sein, wenn Sie alle untergeordneten Elemente einer Liste in einem einzigen Vorgang abrufen möchten, anstatt auf zusätzliche Ereignisse für hinzugefügte untergeordnete Elemente zu warten.
Daten sortieren und filtern
Mit der Realtime Database FIRDatabaseQuery-Klasse können Sie Daten abrufen, die nach Schlüssel, Wert oder dem Wert eines untergeordneten Elements sortiert sind. Sie können das sortierte Ergebnis auch auf eine bestimmte Anzahl von Ergebnissen oder einen Bereich von Schlüsseln oder Werten filtern.
Daten sortieren
Um sortierte Daten abzurufen, geben Sie zuerst eine der Sortiermethoden an, um die Reihenfolge der Ergebnisse zu bestimmen:
| Methode | Nutzung |
|---|---|
queryOrderedByKey
| Ergebnisse nach untergeordneten Schlüsseln sortieren. |
queryOrderedByValue |
Ergebnisse nach untergeordneten Werten sortieren. |
queryOrderedByChild |
Ergebnisse nach dem Wert eines angegebenen untergeordneten Schlüssels oder eines verschachtelten untergeordneten Pfads sortieren. |
Sie können jeweils nur eine Sortiermethode verwenden. Wenn Sie in derselben Abfrage mehrere Sortiermethoden aufrufen, wird ein Fehler ausgelöst.
Das folgende Beispiel zeigt, wie Sie eine Liste der Top-Beiträge eines Nutzers nach der Anzahl der Sterne sortieren können:
Swift
// My top posts by number of stars let myTopPostsQuery = ref.child("user-posts").child(getUid()).queryOrdered(byChild: "starCount")
Objective-C
// My top posts by number of stars FIRDatabaseQuery *myTopPostsQuery = [[[self.ref child:@"user-posts"] child:[super getUid]] queryOrderedByChild:@"starCount"];
Diese Abfrage ruft die Beiträge des Nutzers aus dem Pfad in der Datenbank anhand seiner Nutzer-ID ab, sortiert nach der Anzahl der Sterne, die jeder Beitrag erhalten hat. Diese Technik, IDs als Indexschlüssel zu verwenden, wird als Datenverteilung bezeichnet. Weitere Informationen finden Sie unter Datenbank strukturieren.
Der Aufruf der Methode queryOrderedByChild gibt den untergeordneten Schlüssel an, nach dem die Ergebnisse sortiert werden sollen. In diesem Beispiel werden Beiträge nach dem Wert des
"starCount" untergeordneten Elements in jedem Beitrag sortiert. Abfragen können auch nach verschachtelten untergeordneten Elementen sortiert werden, wenn Ihre Daten so aussehen:
"posts": {
"ts-functions": {
"metrics": {
"views" : 1200000,
"likes" : 251000,
"shares": 1200,
},
"title" : "Why you should use TypeScript for writing Cloud Functions",
"author": "Doug",
},
"android-arch-3": {
"metrics": {
"views" : 900000,
"likes" : 117000,
"shares": 144,
},
"title" : "Using Android Architecture Components with Firebase Realtime Database (Part 3)",
"author": "Doug",
}
},In diesem Fall können wir unsere Listenelemente nach Werten sortieren, die unter dem Schlüssel metrics verschachtelt sind, indem wir den relativen Pfad zum verschachtelten untergeordneten Element in unserem queryOrderedByChild-Aufruf angeben.
Swift
let postsByMostPopular = ref.child("posts").queryOrdered(byChild: "metrics/views")
Objective-C
FIRDatabaseQuery *postsByMostPopular = [[ref child:@"posts"] queryOrderedByChild:@"metrics/views"];
Weitere Informationen zur Sortierung anderer Datentypen finden Sie unter So werden Abfragedaten sortiert.
Daten filtern
Um Daten zu filtern, können Sie beim Erstellen einer Abfrage eine der Methoden für Limit oder Bereich mit einer Sortiermethode kombinieren.
| Methode | Nutzung |
|---|---|
queryLimitedToFirst |
Legt die maximale Anzahl der Elemente fest, die ab dem Anfang der sortierten Liste der Ergebnisse zurückgegeben werden sollen. |
queryLimitedToLast |
Legt die maximale Anzahl der Elemente fest, die ab dem Ende der sortierten Liste der Ergebnisse zurückgegeben werden sollen. |
queryStartingAtValue |
Gibt Elemente zurück, die größer oder gleich dem angegebenen Schlüssel oder Wert sind, je nach gewählter Sortiermethode. |
queryStartingAfterValue |
Gibt Elemente zurück, die größer als der angegebene Schlüssel oder Wert sind, je nach gewählter Sortiermethode. |
queryEndingAtValue |
Gibt Elemente zurück, die kleiner oder gleich dem angegebenen Schlüssel oder Wert sind, je nach gewählter Sortiermethode. |
queryEndingBeforeValue |
Gibt Elemente zurück, die kleiner als der angegebene Schlüssel oder Wert sind, je nach gewählter Sortiermethode. |
queryEqualToValue |
Gibt Elemente zurück, die gleich dem angegebenen Schlüssel oder Wert sind, je nach der gewählten Sortiermethode. |
Im Gegensatz zu den Sortiermethoden können Sie mehrere Limit- oder Bereichsfunktionen kombinieren.
Sie können beispielsweise die Methoden queryStartingAtValue und queryEndingAtValue kombinieren, um die Ergebnisse auf einen bestimmten Wertebereich zu beschränken.
Anzahl der Ergebnisse begrenzen
Mit den Methoden queryLimitedToFirst und queryLimitedToLast können Sie eine maximale Anzahl von untergeordneten Elementen festlegen, die für einen bestimmten Callback synchronisiert werden sollen. Wenn Sie beispielsweise mit queryLimitedToFirst ein Limit von 100 festlegen, erhalten Sie zunächst nur bis zu 100 FIRDataEventTypeChildAdded-Callbacks. Wenn in Ihrer Firebase-Datenbank weniger als 100 Elemente gespeichert sind, wird für jedes Element ein FIRDataEventTypeChildAdded-Callback ausgelöst.
Wenn sich Elemente ändern, erhalten Sie FIRDataEventTypeChildAdded-Callbacks für Elemente, die in die Abfrage aufgenommen werden, und FIRDataEventTypeChildRemoved-Callbacks für Elemente, die aus der Abfrage entfernt werden, sodass die Gesamtzahl bei 100 bleibt.
Das folgende Beispiel zeigt, wie eine Blogging-App eine Liste der 100 neuesten Beiträge aller Nutzer abrufen kann:
Swift
// Last 100 posts, these are automatically the 100 most recent // due to sorting by push() keys let recentPostsQuery = (ref?.child("posts").queryLimited(toFirst: 100))!
Objective-C
// Last 100 posts, these are automatically the 100 most recent // due to sorting by push() keys FIRDatabaseQuery *recentPostsQuery = [[self.ref child:@"posts"] queryLimitedToFirst:100];
Nach Schlüssel oder Wert filtern
Mit queryStartingAtValue, queryStartingAfterValue, queryEndingAtValue, queryEndingBeforeValue und queryEqualToValue können Sie beliebige Start-, End- und Äquivalenzpunkte für Abfragen auswählen. Das kann nützlich sein, um Daten zu paginieren oder Elemente mit untergeordneten Elementen zu finden, die einen bestimmten Wert haben.
So werden Abfragedaten sortiert
In diesem Abschnitt wird erläutert, wie Daten mit den einzelnen Sortiermethoden in der Klasse FIRDatabaseQuery sortiert werden.
queryOrderedByKey
Wenn Sie queryOrderedByKey zum Sortieren Ihrer Daten verwenden, werden die Daten in aufsteigender Reihenfolge nach Schlüssel zurückgegeben.
- Untergeordnete Elemente mit einem Schlüssel, der als 32-Bit-Ganzzahl geparst werden kann, werden zuerst in aufsteigender Reihenfolge sortiert.
- Untergeordnete Elemente mit einem Stringwert als Schlüssel folgen, sortiert in aufsteigender lexikografischer Reihenfolge.
queryOrderedByValue
Wenn Sie queryOrderedByValue verwenden, werden untergeordnete Elemente nach ihrem Wert sortiert. Die Sortierkriterien sind dieselben wie bei queryOrderedByChild, außer dass der Wert des Knotens anstelle des Werts eines angegebenen untergeordneten Schlüssels verwendet wird.
queryOrderedByChild
Wenn Sie queryOrderedByChild verwenden, werden Daten, die den angegebenen untergeordneten Schlüssel enthalten, so sortiert:
- Untergeordnete Elemente mit dem Wert
nilfür den angegebenen untergeordneten Schlüssel werden zuerst angezeigt. - Untergeordnete Elemente mit dem Wert
falsefür den angegebenen untergeordneten Schlüssel kommen als Nächstes. Wenn mehrere untergeordnete Elemente den Wertfalsehaben, werden sie lexikografischsortiert nach Schlüssel. - Untergeordnete Elemente mit dem Wert
truefür den angegebenen untergeordneten Schlüssel kommen als Nächstes. Wenn mehrere untergeordnete Elemente den Werttruehaben, werden sie lexikografisch nach Schlüssel sortiert. - Untergeordnete Elemente mit einem numerischen Wert folgen, sortiert in aufsteigender Reihenfolge. Wenn mehrere untergeordnete Elemente denselben numerischen Wert für den angegebenen untergeordneten Knoten haben, werden sie nach Schlüssel sortiert.
- Strings folgen nach Zahlen und werden in aufsteigender lexikografischer Reihenfolge sortiert. Wenn mehrere untergeordnete Elemente denselben Wert für den angegebenen untergeordneten Knoten haben, werden sie lexikografisch nach Schlüssel sortiert.
- Objekte folgen zuletzt und werden in aufsteigender lexikografischer Reihenfolge nach Schlüssel sortiert.
Listener trennen
Observer beenden die Datensynchronisierung nicht automatisch, wenn Sie einen ViewController verlassen. Wenn ein Observer nicht ordnungsgemäß entfernt wird, synchronisiert er weiterhin Daten mit dem lokalen Speicher und behält alle Objekte bei, die in der Closure des Ereignishandlers erfasst wurden. Das kann zu Speicherlecks führen. Wenn ein Observer nicht mehr benötigt wird, entfernen Sie ihn, indem Sie das zugehörige FIRDatabaseHandle an die Methode removeObserverWithHandle übergeben.
Wenn Sie einer Referenz einen Callback-Block hinzufügen, wird ein FIRDatabaseHandle zurückgegeben.
Mit diesen Handles kann der Callback-Block entfernt werden.
Wenn einer Datenbankreferenz mehrere Listener hinzugefügt wurden, wird jeder Listener aufgerufen, wenn ein Ereignis ausgelöst wird. Um die Datensynchronisierung an diesem Ort zu beenden, müssen Sie alle Observer an einem Ort entfernen, indem Sie die Methode removeAllObservers aufrufen.
Wenn Sie removeObserverWithHandle oder removeAllObservers für einen Listener aufrufen, werden Listener, die für die untergeordneten Knoten registriert sind, nicht automatisch entfernt. Sie müssen diese Referenzen oder Handles auch im Blick behalten, um sie zu entfernen.