Utiliser des listes de données sur les plates-formes Apple

Obtenir une FIRDatabaseReference

Pour lire ou écrire des données dans la base de données, vous avez besoin d'une instance de FIRDatabaseReference :

Swift

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
var ref: DatabaseReference!

ref = Database.database().reference()

Objective-C

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
@property (strong, nonatomic) FIRDatabaseReference *ref;

self.ref = [[FIRDatabase database] reference];

Lire et écrire des listes

Ajouter des données à une liste

Utilisez la méthode childByAutoId pour ajouter des données à une liste dans les applications multi-utilisateurs. La méthode childByAutoId génère une clé unique chaque fois qu'un nouvel enfant est ajouté à la référence Firebase spécifiée. En utilisant ces clés générées automatiquement pour chaque nouvel élément de la liste, plusieurs clients peuvent ajouter des enfants au même emplacement en même temps sans conflit d'écriture. La clé unique générée par childByAutoId est basée sur un code temporel. Les éléments de la liste sont donc automatiquement classés par ordre chronologique.

Vous pouvez utiliser la référence aux nouvelles données renvoyées par la méthode childByAutoId pour obtenir la valeur de la clé générée automatiquement de l'enfant ou définir des données pour l'enfant. L'appel de getKey sur une référence childByAutoId renvoie la clé générée automatiquement.

Vous pouvez utiliser ces clés générées automatiquement pour simplifier l'aplatissement de votre structure de données. Pour en savoir plus, consultez l'exemple de distribution des données .

Écouter les événements enfants

Les événements enfants sont déclenchés en réponse à des opérations spécifiques qui se produisent sur les enfants d'un nœud à partir d'une opération telle qu'un nouvel enfant ajouté via la méthode childByAutoId ou un enfant mis à jour via la méthode updateChildValues.

Type d'événement Utilisation type
FIRDataEventTypeChildAdded Récupérer des listes d'éléments ou écouter les ajouts à une liste d'éléments. Cet événement est déclenché une fois pour chaque enfant existant, puis à chaque fois qu'un nouvel enfant est ajouté au chemin spécifié. L'écouteur reçoit un instantané contenant les données du nouvel enfant.
FIRDataEventTypeChildChanged Écouter les modifications apportées aux éléments d'une liste. Cet événement est déclenché chaque fois qu'un nœud enfant est modifié. Cela inclut toutes les modifications apportées aux descendants du nœud enfant. L'instantané transmis à l'écouteur d'événements contient les données mises à jour de l'enfant.
FIRDataEventTypeChildRemoved Écouter les éléments supprimés d'une liste. Cet événement est déclenché lorsqu' un enfant immédiat est supprimé.L'instantané transmis au bloc de rappel contient les données de l'enfant supprimé.
FIRDataEventTypeChildMoved Écouter les modifications apportées à l'ordre des éléments d'une liste ordonnée. Cet événement est déclenché chaque fois qu'une mise à jour entraîne un réordonnancement de l' enfant. Il est utilisé avec des données ordonnées par queryOrderedByChild ou queryOrderedByValue.

Chacun de ces éléments peut être utile pour écouter les modifications apportées à un nœud spécifique d'une base de données. Par exemple, une application de blog social peut utiliser ces méthodes ensemble pour surveiller l'activité dans les commentaires d'un post, comme illustré ci-dessous :

Swift

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
// 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

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
// 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];
 }];

Écouter les événements de valeur

Bien que l'écoute des événements enfants soit la méthode recommandée pour lire des listes de données, il est parfois utile d'écouter les événements de valeur sur une référence de liste.

L'association d'un observateur FIRDataEventTypeValue à une liste de données renvoie l'intégralité de la liste de données sous la forme d'un seul DataSnapshot, que vous pouvez ensuite parcourir en boucle pour accéder à des enfants individuels.

Même lorsqu'il n'y a qu'une seule correspondance pour la requête, l'instantané reste une liste. Il ne contient qu'un seul élément. Pour accéder à l'élément, vous devez parcourir le résultat en boucle :

Swift

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
_commentsRef.observe(.value) { snapshot in
  for child in snapshot.children {
    ...
  }
}

Objective-C

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
[_commentsRef
              observeEventType:FIRDataEventTypeValue
              withBlock:^(FIRDataSnapshot *snapshot) {
                // Loop over children
                NSEnumerator *children = [snapshot children];
                FIRDataSnapshot *child;
                while (child = [children nextObject]) {
                  // ...
                }
              }];

Ce modèle peut être utile lorsque vous souhaitez récupérer tous les enfants d'une liste en une seule opération, plutôt que d'écouter d'autres événements enfants ajoutés.

Trier et filtrer des données

Vous pouvez utiliser la classe Realtime Database FIRDatabaseQuery pour récupérer des données triées par clé, par valeur ou par la valeur d'un enfant. Vous pouvez également filtrer le résultat trié sur un nombre spécifique de résultats ou une plage de clés ou de valeurs.

Trier les données

Pour récupérer des données triées, commencez par spécifier l'une des méthodes de tri afin de déterminer l'ordre des résultats :

Méthode Utilisation
queryOrderedByKey Trier les résultats par clés enfants.
queryOrderedByValue Trier les résultats par valeurs enfants.
queryOrderedByChild Trier les résultats par la valeur d'une clé enfant spécifiée ou d'un chemin enfant imbriqué.

Vous ne pouvez utiliser qu'une seule méthode de tri à la fois. L'appel d'une méthode de tri plusieurs fois dans la même requête génère une erreur.

L'exemple suivant montre comment récupérer une liste des meilleurs posts d'un utilisateur, triés par nombre d'étoiles :

Swift

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
// My top posts by number of stars
let myTopPostsQuery = ref.child("user-posts").child(getUid()).queryOrdered(byChild: "starCount")

Objective-C

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
// My top posts by number of stars
FIRDatabaseQuery *myTopPostsQuery = [[[self.ref child:@"user-posts"]
                                      child:[super getUid]]
                                     queryOrderedByChild:@"starCount"];

Cette requête récupère les posts de l'utilisateur à partir du chemin d'accès dans la base de données en fonction de son ID utilisateur, triés par le nombre d'étoiles reçues par chaque post. Cette technique d'utilisation des ID comme clés d'index est appelée distribution des données. Pour en savoir plus, consultez la section Structurer votre base de données.

L'appel à la méthode queryOrderedByChild spécifie la clé enfant par laquelle trier les résultats. Dans cet exemple, les posts sont triés par la valeur de l' "starCount" enfant dans chaque post. Les requêtes peuvent également être triées par des enfants imbriqués, si vos données se présentent comme suit :

"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",
  }
},

Dans ce cas, nous pouvons trier les éléments de notre liste par valeurs imbriquées sous la clé metrics en spécifiant le chemin relatif vers l'enfant imbriqué dans notre appel queryOrderedByChild.

Swift

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
 
let postsByMostPopular = ref.child("posts").queryOrdered(byChild: "metrics/views")

Objective-C

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
 
FIRDatabaseQuery *postsByMostPopular = [[ref child:@"posts"] queryOrderedByChild:@"metrics/views"];

Pour en savoir plus sur l'ordre des autres types de données, consultez la section Ordre des données de requête.

Filtrer les données

Pour filtrer les données, vous pouvez combiner l'une des méthodes de limite ou de plage avec une méthode de tri lors de la construction d'une requête.

Méthode Utilisation
queryLimitedToFirst Définit le nombre maximal d'éléments à renvoyer depuis le début de la liste ordonnée des résultats.
queryLimitedToLast Définit le nombre maximal d'éléments à renvoyer depuis la fin de la liste ordonnée des résultats.
queryStartingAtValue Renvoie les éléments supérieurs ou égaux à la clé ou à la valeur spécifiée, selon la méthode de tri choisie.
queryStartingAfterValue Renvoie les éléments supérieurs à la clé ou à la valeur spécifiée, selon la méthode de tri choisie.
queryEndingAtValue Renvoie les éléments inférieurs ou égaux à la clé ou à la valeur spécifiée, selon la méthode de tri choisie.
queryEndingBeforeValue Renvoie les éléments inférieurs à la clé ou à la valeur spécifiée, selon la méthode de tri choisie.
queryEqualToValue Renvoie les éléments égaux à la clé ou à la valeur spécifiée, selon la méthode de tri choisie.

Contrairement aux méthodes de tri, vous pouvez combiner plusieurs fonctions de limite ou de plage. Par exemple, vous pouvez combiner les méthodes queryStartingAtValue et queryEndingAtValue pour limiter les résultats à une plage de valeurs spécifiée.

Limiter le nombre de résultats

Vous pouvez utiliser les méthodes queryLimitedToFirst et queryLimitedToLast pour définir un nombre maximal d'enfants à synchroniser pour un rappel donné. Par exemple, si vous utilisez queryLimitedToFirst pour définir une limite de 100, vous ne recevez initialement que 100 rappels FIRDataEventTypeChildAdded au maximum. Si vous avez moins de 100 éléments stockés dans votre base de données Firebase, un rappel FIRDataEventTypeChildAdded est déclenché pour chaque élément.

Lorsque des éléments changent, vous recevez des rappels FIRDataEventTypeChildAdded pour les éléments qui entrent dans la requête et des rappels FIRDataEventTypeChildRemoved pour les éléments qui en sortent, de sorte que le nombre total reste à 100.

L'exemple suivant montre comment une application de blog peut récupérer une liste des 100 posts les plus récents de tous les utilisateurs :

Swift

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
// 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

Remarque : Ce produit Firebase n'est pas disponible sur la cible App Clip.
// Last 100 posts, these are automatically the 100 most recent
// due to sorting by push() keys
FIRDatabaseQuery *recentPostsQuery = [[self.ref child:@"posts"] queryLimitedToFirst:100];

Filtrer par clé ou valeur

Vous pouvez utiliser queryStartingAtValue, queryStartingAfterValue, queryEndingAtValue, queryEndingBeforeValue et queryEqualToValue pour choisir des points de départ, de fin et d'équivalence arbitraires pour les requêtes. Cela peut être utile pour paginer des données ou trouver des éléments avec des enfants ayant une valeur spécifique.

Ordre des données de requête

Cette section explique comment les données sont triées par chacune des méthodes de tri de la classe FIRDatabaseQuery.

queryOrderedByKey

Lorsque vous utilisez queryOrderedByKey pour trier vos données, celles-ci sont renvoyées par ordre croissant de clé.

  1. Les enfants dont la clé peut être analysée comme un entier de 32 bits sont affichés en premier, triés par ordre croissant.
  2. Les enfants dont la clé est une valeur de chaîne sont affichés ensuite, triés par ordre lexicographique croissant.

queryOrderedByValue

Lorsque vous utilisez queryOrderedByValue, les enfants sont triés par leur valeur. Les critères de tri sont les mêmes que dans queryOrderedByChild, sauf que la valeur du nœud est utilisée à la place de la valeur d'une clé enfant spécifiée.

queryOrderedByChild

Lorsque vous utilisez queryOrderedByChild, les données contenant la clé enfant spécifiée sont triées comme suit :

  1. Les enfants dont la valeur de la clé enfant spécifiée est nil sont affichés en premier.
  2. Les enfants dont la valeur de la clé enfant spécifiée est false sont affichés ensuite. Si plusieurs enfants ont la valeur false, ils sont triés par ordre lexicographique de clé.
  3. Les enfants dont la valeur de la clé enfant spécifiée est true sont affichés ensuite. Si plusieurs enfants ont la valeur true, ils sont triés par ordre lexicographique de clé.
  4. Les enfants dont la valeur est numérique sont affichés ensuite, triés par ordre croissant. Si plusieurs enfants ont la même valeur numérique pour le nœud enfant spécifié, ils sont triés par clé.
  5. Les chaînes sont affichées après les nombres et triées par ordre lexicographique croissant. Si plusieurs enfants ont la même valeur pour le nœud enfant spécifié, ils sont triés par ordre lexicographique de clé.
  6. Les objets sont affichés en dernier et triés par ordre lexicographique de clé.

Détacher les écouteurs

Les observateurs n'arrêtent pas automatiquement la synchronisation des données lorsque vous quittez un ViewController. Si un observateur n'est pas correctement supprimé, il continue de synchroniser les données dans la mémoire locale et conserve tous les objets capturés dans la fermeture du gestionnaire d'événements, ce qui peut entraîner des fuites de mémoire. Lorsqu'un observateur n'est plus nécessaire, supprimez-le en transmettant le FIRDatabaseHandle associé à la méthode removeObserverWithHandle.

Lorsque vous ajoutez un bloc de rappel à une référence, un FIRDatabaseHandle est renvoyé. Ces handles peuvent être utilisés pour supprimer le bloc de rappel.

Si plusieurs écouteurs ont été ajoutés à une référence de base de données, chaque écouteur est appelé lorsqu'un événement est déclenché. Pour arrêter la synchronisation des données à cet emplacement, vous devez supprimer tous les observateurs à un emplacement en appelant la méthode removeAllObservers.

L'appel de removeObserverWithHandle ou removeAllObservers sur un écouteur ne supprime pas automatiquement les écouteurs enregistrés sur ses nœuds enfants. Vous devez également suivre ces références ou handles pour les supprimer.

Étapes suivantes