Làm việc với Danh sách dữ liệu trên các nền tảng của Apple

Sử dụng bộ sưu tập để sắp xếp ngăn nắp các trang Lưu và phân loại nội dung dựa trên lựa chọn ưu tiên của bạn.

Nhận một FIRDatabaseReference

Để đọc hoặc ghi dữ liệu từ cơ sở dữ liệu, bạn cần một phiên bản của FIRDatabaseReference :

Nhanh

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
var ref: DatabaseReference!

ref = Database.database().reference()

Mục tiêu-C

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
@property (strong, nonatomic) FIRDatabaseReference *ref;

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

Đọc và viết danh sách

Nối vào danh sách dữ liệu

Sử dụng phương thức childByAutoId để thêm dữ liệu vào danh sách trong các ứng dụng nhiều người dùng. Phương thức childByAutoId tạo một khóa duy nhất mỗi khi một phần tử con mới được thêm vào tham chiếu Firebase đã chỉ định. Bằng cách sử dụng các khóa được tạo tự động này cho từng thành phần mới trong danh sách, một số khách hàng có thể thêm phần tử con vào cùng một vị trí cùng một lúc mà không có xung đột ghi. Khóa duy nhất được tạo bởi childByAutoId dựa trên dấu thời gian, vì vậy các mục trong danh sách được tự động sắp xếp theo trình tự thời gian.

Bạn có thể sử dụng tham chiếu đến dữ liệu mới do phương thức childByAutoId trả về để lấy giá trị của khóa được tạo tự động của trẻ hoặc dữ liệu đã đặt cho trẻ. Gọi getKey trên tham chiếu childByAutoId trả về khóa được tạo tự động.

Bạn có thể sử dụng các khóa được tạo tự động này để đơn giản hóa việc làm phẳng cấu trúc dữ liệu của mình. Để biết thêm thông tin, hãy xem ví dụ về phân xuất dữ liệu .

Lắng nghe các sự kiện con

Các sự kiện con được kích hoạt để đáp ứng với các hoạt động cụ thể xảy ra với các nút con của một nút từ một hoạt động, chẳng hạn như một phần tử con mới được thêm thông qua phương thức childByAutoId hoặc một phần tử con được cập nhật thông qua phương thức updateChildValues .

Loại sự kiện sử dụng điển hình
FIRDataEventTypeChildAdded Truy xuất danh sách các mục hoặc lắng nghe các bổ sung cho danh sách các mục. Sự kiện này được kích hoạt một lần cho mỗi phần tử con hiện có và sau đó được kích hoạt lại mỗi khi một phần tử con mới được thêm vào đường dẫn đã chỉ định. Người nghe được chuyển một ảnh chụp nhanh có chứa dữ liệu của đứa trẻ mới.
FIRDataEventTypeChildChanged Lắng nghe những thay đổi đối với các mục trong danh sách. Sự kiện này được kích hoạt bất cứ khi nào nút con được sửa đổi. Điều này bao gồm bất kỳ sửa đổi nào đối với hậu duệ của nút con. Ảnh chụp nhanh được chuyển đến trình xử lý sự kiện chứa dữ liệu được cập nhật cho phần tử con.
FIRDataEventTypeChildRemoved Lắng nghe các mục bị xóa khỏi danh sách. Sự kiện này được kích hoạt khi một phần tử con ngay lập tức bị xóa. Ảnh chụp nhanh được chuyển đến khối gọi lại chứa dữ liệu cho phần tử con đã bị xóa.
FIRDataEventTypeChildMoved Lắng nghe những thay đổi về thứ tự của các mục trong danh sách có thứ tự. Sự kiện này được kích hoạt bất cứ khi nào một bản cập nhật gây ra sự sắp xếp lại thứ tự của đứa trẻ. Nó được sử dụng với dữ liệu được sắp xếp theo queryOrderedByChild hoặc queryOrderedByValue .

Mỗi thứ này cùng nhau có thể hữu ích để lắng nghe các thay đổi đối với một nút cụ thể trong cơ sở dữ liệu. Ví dụ: một ứng dụng viết blog xã hội có thể sử dụng các phương pháp này cùng nhau để theo dõi hoạt động trong phần nhận xét của bài đăng, như minh họa bên dưới:

Nhanh

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
// 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
  )
})

Mục tiêu-C

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
// 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];
 }];

Lắng nghe các sự kiện giá trị

Mặc dù lắng nghe các sự kiện con là cách được khuyến nghị để đọc danh sách dữ liệu, nhưng có những trường hợp lắng nghe các sự kiện giá trị trên danh sách tham chiếu là hữu ích.

Việc đính kèm một trình quan sát FIRDataEventTypeValue vào danh sách dữ liệu sẽ trả về toàn bộ danh sách dữ liệu dưới dạng một Ảnh chụp nhanh dữ liệu duy nhất, sau đó bạn có thể lặp lại để truy cập từng phần tử con.

Ngay cả khi chỉ có một kết quả phù hợp duy nhất cho truy vấn, ảnh chụp nhanh vẫn là một danh sách; nó chỉ chứa một mục duy nhất. Để truy cập mục này, bạn cần lặp lại kết quả:

Nhanh

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
_commentsRef.observe(.value) { snapshot in
  for child in snapshot.children {
    ...
  }
}

Mục tiêu-C

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
[_commentsRef
              observeEventType:FIRDataEventTypeValue
              withBlock:^(FIRDataSnapshot *snapshot) {
                // Loop over children
                NSEnumerator *children = [snapshot children];
                FIRDataSnapshot *child;
                while (child = [children nextObject]) {
                  // ...
                }
              }];

Mẫu này có thể hữu ích khi bạn muốn tìm nạp tất cả phần tử con của danh sách trong một thao tác đơn lẻ, thay vì lắng nghe thêm các sự kiện phần tử con được thêm vào.

Sắp xếp và lọc dữ liệu

Bạn có thể sử dụng lớp FIRDatabaseQuery của Cơ sở dữ liệu thời gian thực để truy xuất dữ liệu được sắp xếp theo khóa, theo giá trị hoặc theo giá trị của con. Bạn cũng có thể lọc kết quả đã sắp xếp thành một số kết quả cụ thể hoặc một dải khóa hoặc giá trị.

Sắp xếp dữ liệu

Để truy xuất dữ liệu đã sắp xếp, hãy bắt đầu bằng cách chỉ định một trong các phương pháp sắp xếp theo thứ tự để xác định cách sắp xếp kết quả:

Phương pháp Cách sử dụng
queryOrderedByKey Thứ tự kết quả theo các phím con.
queryOrderedByValue Sắp xếp thứ tự kết quả theo các giá trị con.
queryOrderedByChild Sắp xếp kết quả theo giá trị của khóa con được chỉ định hoặc đường dẫn con lồng nhau.

Bạn chỉ có thể sử dụng một phương thức đặt hàng tại một thời điểm. Gọi một phương thức theo thứ tự nhiều lần trong cùng một truy vấn sẽ gây ra lỗi.

Ví dụ sau minh họa cách bạn có thể truy xuất danh sách các bài đăng hàng đầu của người dùng được sắp xếp theo số sao của họ:

Nhanh

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
// My top posts by number of stars
let myTopPostsQuery = ref.child("user-posts").child(getUid()).queryOrdered(byChild: "starCount")

Mục tiêu-C

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
// My top posts by number of stars
FIRDatabaseQuery *myTopPostsQuery = [[[self.ref child:@"user-posts"]
                                      child:[super getUid]]
                                     queryOrderedByChild:@"starCount"];

Truy vấn này truy xuất các bài đăng của người dùng từ đường dẫn trong cơ sở dữ liệu dựa trên ID người dùng của họ, được sắp xếp theo số sao mà mỗi bài đăng nhận được. Kỹ thuật sử dụng ID làm khóa chỉ mục này được gọi là phân bổ dữ liệu, bạn có thể đọc thêm về kỹ thuật này trong Cấu trúc cơ sở dữ liệu của bạn .

Lệnh gọi phương thức queryOrderedByChild chỉ định khóa con để sắp xếp kết quả theo. Trong ví dụ này, các bài đăng được sắp xếp theo giá trị của phần tử con "starCount" trong mỗi bài đăng. Các truy vấn cũng có thể được sắp xếp theo các phần tử con lồng nhau, trong trường hợp bạn có dữ liệu giống như sau:

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

Trong trường hợp này, chúng tôi có thể sắp xếp thứ tự các phần tử danh sách của mình theo các giá trị được lồng trong khóa metrics bằng cách chỉ định đường dẫn tương đối tới phần tử con được lồng trong lệnh gọi queryOrderedByChild của chúng tôi.

Nhanh

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
 
let postsByMostPopular = ref.child("posts").queryOrdered(byChild: "metrics/views")

Mục tiêu-C

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
 
FIRDatabaseQuery *postsByMostPopular = [[ref child:@"posts"] queryOrderedByChild:@"metrics/views"];

Để biết thêm thông tin về cách sắp xếp các loại dữ liệu khác, hãy xem Cách sắp xếp dữ liệu truy vấn .

Lọc dữ liệu

Để lọc dữ liệu, bạn có thể kết hợp bất kỳ phương pháp giới hạn hoặc phạm vi nào với phương pháp sắp xếp theo thứ tự khi tạo truy vấn.

Phương pháp Cách sử dụng
queryLimitedToFirst Đặt số lượng mục tối đa để trả về từ đầu danh sách kết quả được sắp xếp.
queryLimitedToLast Đặt số lượng mục tối đa sẽ trả về từ cuối danh sách kết quả được sắp xếp.
queryStartingAtValue Trả về các mục lớn hơn hoặc bằng khóa hoặc giá trị đã chỉ định, tùy thuộc vào phương thức đặt hàng đã chọn.
queryStartingAfterValue Trả về các mục lớn hơn khóa hoặc giá trị đã chỉ định, tùy thuộc vào phương thức đặt hàng đã chọn.
queryEndingAtValue Trả lại các mục nhỏ hơn hoặc bằng khóa hoặc giá trị đã chỉ định, tùy thuộc vào phương thức đặt hàng đã chọn.
queryEndingBeforeValue Trả lại các mục ít hơn khóa hoặc giá trị đã chỉ định, tùy thuộc vào phương thức đặt hàng đã chọn.
queryEqualToValue Trả về các mục bằng với khóa hoặc giá trị đã chỉ định, tùy thuộc vào phương thức đặt hàng đã chọn.

Không giống như các phương pháp sắp xếp theo thứ tự, bạn có thể kết hợp nhiều hàm giới hạn hoặc phạm vi. Ví dụ: bạn có thể kết hợp các phương thức queryStartingAtValuequeryEndingAtValue để giới hạn kết quả trong phạm vi giá trị đã chỉ định.

Giới hạn số lượng kết quả

Bạn có thể sử dụng các phương thức queryLimitedToFirstqueryLimitedToLast để đặt số lượng phần tử con tối đa được đồng bộ hóa cho một cuộc gọi lại nhất định. Ví dụ: nếu bạn sử dụng queryLimitedToFirst để đặt giới hạn là 100, ban đầu bạn chỉ nhận được tối đa 100 lệnh gọi lại FIRDataEventTypeChildAdded . Nếu bạn có ít hơn 100 mục được lưu trữ trong cơ sở dữ liệu Firebase của mình, thì lệnh gọi lại FIRDataEventTypeChildAdded kích hoạt cho từng mục.

Khi các mục thay đổi, bạn nhận được lệnh gọi lại FIRDataEventTypeChildAdded cho các mục nhập truy vấn và lệnh gọi lại FIRDataEventTypeChildRemoved cho các mục bị loại khỏi truy vấn để tổng số vẫn ở mức 100.

Ví dụ sau minh họa cách một ứng dụng viết blog mẫu có thể truy xuất danh sách 100 bài đăng gần đây nhất của tất cả người dùng:

Nhanh

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
// Last 100 posts, these are automatically the 100 most recent
// due to sorting by push() keys
let recentPostsQuery = (ref?.child("posts").queryLimited(toFirst: 100))!

Mục tiêu-C

Lưu ý: Sản phẩm Firebase này không khả dụng trên mục tiêu Clip ứng dụng.
// Last 100 posts, these are automatically the 100 most recent
// due to sorting by push() keys
FIRDatabaseQuery *recentPostsQuery = [[self.ref child:@"posts"] queryLimitedToFirst:100];

Lọc theo khóa hoặc giá trị

Bạn có thể sử dụng queryStartingAtValue , queryStartingAfterValue , queryEndingAtValue , queryEndingBeforeValuequeryEqualToValue để chọn điểm bắt đầu, điểm kết thúc và điểm tương đương tùy ý cho các truy vấn. Điều này có thể hữu ích để phân trang dữ liệu hoặc tìm các mục có con có giá trị cụ thể.

Cách dữ liệu truy vấn được sắp xếp

Phần này giải thích cách dữ liệu được sắp xếp theo từng phương thức theo thứ tự trong lớp FIRDatabaseQuery .

queryOrderedByKey

Khi sử dụng queryOrderedByKey để sắp xếp dữ liệu của bạn, dữ liệu được trả về theo thứ tự tăng dần theo khóa.

  1. Phần tử con có khóa có thể được phân tích cú pháp dưới dạng số nguyên 32 bit sẽ đến trước, được sắp xếp theo thứ tự tăng dần.
  2. Trẻ em có giá trị chuỗi làm khóa tiếp theo, được sắp xếp theo từ điển theo thứ tự tăng dần.

queryOrderedByValue

Khi sử dụng queryOrderedByValue , trẻ em được sắp xếp theo giá trị của chúng. Tiêu chí đặt hàng giống như trong queryOrderedByChild , ngoại trừ giá trị của nút được sử dụng thay vì giá trị của khóa con đã chỉ định.

queryOrderedByChild

Khi sử dụng queryOrderedByChild , dữ liệu chứa khóa con đã chỉ định được sắp xếp như sau:

  1. Trẻ em có giá trị nil cho khóa con được chỉ định đến trước.
  2. Các phần tử con có giá trị false cho khóa con được chỉ định sẽ xuất hiện tiếp theo. Nếu nhiều phần tử con có giá trị false , chúng sẽ được sắp xếp theo từ điển theo khóa.
  3. Tiếp theo là các phần tử con có giá trị true cho khóa con được chỉ định. Nếu nhiều phần tử con có giá trị true , thì chúng được sắp xếp theo từ điển theo khóa.
  4. Trẻ em có giá trị số tiếp theo, được sắp xếp theo thứ tự tăng dần. Nếu nhiều nút con có cùng giá trị số cho nút con đã chỉ định, chúng sẽ được sắp xếp theo khóa.
  5. Các chuỗi đứng sau các số và được sắp xếp theo từ điển theo thứ tự tăng dần. Nếu nhiều nút con có cùng giá trị cho nút con đã chỉ định, chúng sẽ được sắp xếp theo thứ tự từ điển theo khóa.
  6. Các đối tượng đến cuối cùng và được sắp xếp theo từ điển theo khóa theo thứ tự tăng dần.

Tách người nghe

Người quan sát không tự động dừng đồng bộ hóa dữ liệu khi bạn rời khỏi ViewController . Nếu một trình quan sát không bị xóa đúng cách, nó sẽ tiếp tục đồng bộ hóa dữ liệu với bộ nhớ cục bộ và sẽ giữ lại bất kỳ đối tượng nào được ghi lại trong quá trình đóng trình xử lý sự kiện, điều này có thể gây rò rỉ bộ nhớ. Khi một người quan sát không còn cần thiết, hãy loại bỏ nó bằng cách chuyển FIRDatabaseHandle được liên kết đến phương thức removeObserverWithHandle .

Khi bạn thêm một khối gọi lại vào một tham chiếu, FIRDatabaseHandle sẽ được trả về. Các tay cầm này có thể được sử dụng để loại bỏ khối gọi lại.

Nếu nhiều người nghe đã được thêm vào một tham chiếu cơ sở dữ liệu, thì mỗi người nghe sẽ được gọi khi một sự kiện được nêu ra. Để dừng đồng bộ hóa dữ liệu tại vị trí đó, bạn phải xóa tất cả người quan sát tại một vị trí bằng cách gọi phương thức removeAllObservers .

Gọi removeObserverWithHandle hoặc removeAllObservers trên một trình nghe không tự động xóa các trình nghe đã đăng ký trên các nút con của nó; bạn cũng phải theo dõi các tài liệu tham khảo hoặc xử lý đó để xóa chúng.

Bước tiếp theo