Tài liệu này trình bày cách thao tác với danh sách dữ liệu trong Firebase. Để tìm hiểu những kiến thức cơ bản về cách đọc và ghi dữ liệu Firebase, hãy xem phần Đọc và ghi dữ liệu trên Android.
Lấy DatabaseReference
Để đọc và ghi dữ liệu từ cơ sở dữ liệu, bạn cần có một thực thể DatabaseReference
:
Kotlin
private lateinit var database: DatabaseReference // ... database = Firebase.database.reference
Java
private DatabaseReference mDatabase; // ... mDatabase = FirebaseDatabase.getInstance().getReference();
Đọc và ghi danh sách
Thêm vào danh sách dữ liệu
Sử dụng phương thức push()
để 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 push()
sẽ tạo một khoá 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 khoá được tạo tự động này cho mỗi phần tử mới trong danh sách, một số ứng dụng có thể thêm các phần tử con vào cùng một vị trí cùng lúc mà không xảy ra xung đột ghi. Khoá duy nhất do push()
tạo dựa trên dấu thời gian, vì vậy, các mục trong danh sách sẽ tự động được sắp xếp theo trình tự thời gian.
Bạn có thể dùng thông tin tham chiếu đến dữ liệu mới do phương thức push()
trả về để lấy giá trị của khoá được tạo tự động của đối tượng con hoặc đặt dữ liệu cho đối tượng con. Việc gọi getKey()
trên một giá trị tham chiếu push()
sẽ trả về giá trị của khoá được tạo tự động.
Bạn có thể sử dụng các khoá được tạo tự động này để đơn giản hoá việc làm phẳng cấu trúc dữ liệu. Để biết thêm thông tin, hãy xem ví dụ về việc phân phối dữ liệu.
Theo dõi các sự kiện phụ
Khi làm việc với danh sách, ứng dụng của bạn nên theo dõi các sự kiện con thay vì các sự kiện giá trị được dùng cho các đối tượng đơn lẻ.
Các sự kiện con được kích hoạt để phản hồi các thao tác cụ thể xảy ra với các thành phần con của một nút từ một thao tác, chẳng hạn như một thành phần con mới được thêm thông qua phương thức push()
hoặc một thành phần con được cập nhật thông qua phương thức updateChildren()
.
Mỗi phương thức này có thể hữu ích khi theo dõi các thay đổi đối với một nút cụ thể trong cơ sở dữ liệu.
Để theo dõi các sự kiện con trên DatabaseReference
, hãy đính kèm một ChildEventListener
:
gửi biểu mẫu | Lệnh gọi lại sự kiện | Mức sử dụng thông thường |
---|---|---|
ChildEventListener
| onChildAdded() |
Truy xuất danh sách các mục hoặc theo dõi các mục được thêm vào danh sách.
Lệnh gọi lại này được kích hoạt một lần cho mỗi thành phần con hiện có, sau đó kích hoạt lại mỗi khi một thành phần con mới được thêm vào đường dẫn đã chỉ định. DataSnapshot được truyền đến trình nghe chứa dữ liệu của trẻ em mới.
|
onChildChanged() |
Theo dõi các thay đổi đối với các mục trong danh sách. Sự kiện này sẽ kích hoạt bất cứ khi nào một nút con được sửa đổi, bao gồm cả mọi nội dung sửa đổi đối với các nút con của nút con đó. DataSnapshot được truyền đến trình nghe sự kiện chứa dữ liệu đã cập nhật cho thành phần con.
|
|
onChildRemoved() |
Lắng nghe các mục bị xoá khỏi danh sách. DataSnapshot được truyền đến lệnh gọi lại sự kiện chứa dữ liệu cho trẻ em đã bị xoá.
|
|
onChildMoved() |
Lắng nghe các 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 lệnh gọi lại onChildChanged() được kích hoạt bởi một bản cập nhật gây ra việc sắp xếp lại thành phần con.
Phương thức này được dùng với dữ liệu được sắp xếp bằng orderByChild hoặc orderByValue .
|
Ví dụ: một ứng dụng blog xã hội có thể sử dụng đồng thời các phương thức này để theo dõi hoạt động trong phần bình luận của một bài đăng, như minh hoạ dưới đây:
Kotlin
val childEventListener = object : ChildEventListener { override fun onChildAdded(dataSnapshot: DataSnapshot, previousChildName: String?) { Log.d(TAG, "onChildAdded:" + dataSnapshot.key!!) // A new comment has been added, add it to the displayed list val comment = dataSnapshot.getValue<Comment>() // ... } override fun onChildChanged(dataSnapshot: DataSnapshot, previousChildName: String?) { Log.d(TAG, "onChildChanged: ${dataSnapshot.key}") // A comment has changed, use the key to determine if we are displaying this // comment and if so displayed the changed comment. val newComment = dataSnapshot.getValue<Comment>() val commentKey = dataSnapshot.key // ... } override fun onChildRemoved(dataSnapshot: DataSnapshot) { Log.d(TAG, "onChildRemoved:" + dataSnapshot.key!!) // A comment has changed, use the key to determine if we are displaying this // comment and if so remove it. val commentKey = dataSnapshot.key // ... } override fun onChildMoved(dataSnapshot: DataSnapshot, previousChildName: String?) { Log.d(TAG, "onChildMoved:" + dataSnapshot.key!!) // A comment has changed position, use the key to determine if we are // displaying this comment and if so move it. val movedComment = dataSnapshot.getValue<Comment>() val commentKey = dataSnapshot.key // ... } override fun onCancelled(databaseError: DatabaseError) { Log.w(TAG, "postComments:onCancelled", databaseError.toException()) Toast.makeText( context, "Failed to load comments.", Toast.LENGTH_SHORT, ).show() } } databaseReference.addChildEventListener(childEventListener)
Java
ChildEventListener childEventListener = new ChildEventListener() { @Override public void onChildAdded(DataSnapshot dataSnapshot, String previousChildName) { Log.d(TAG, "onChildAdded:" + dataSnapshot.getKey()); // A new comment has been added, add it to the displayed list Comment comment = dataSnapshot.getValue(Comment.class); // ... } @Override public void onChildChanged(DataSnapshot dataSnapshot, String previousChildName) { Log.d(TAG, "onChildChanged:" + dataSnapshot.getKey()); // A comment has changed, use the key to determine if we are displaying this // comment and if so displayed the changed comment. Comment newComment = dataSnapshot.getValue(Comment.class); String commentKey = dataSnapshot.getKey(); // ... } @Override public void onChildRemoved(DataSnapshot dataSnapshot) { Log.d(TAG, "onChildRemoved:" + dataSnapshot.getKey()); // A comment has changed, use the key to determine if we are displaying this // comment and if so remove it. String commentKey = dataSnapshot.getKey(); // ... } @Override public void onChildMoved(DataSnapshot dataSnapshot, String previousChildName) { Log.d(TAG, "onChildMoved:" + dataSnapshot.getKey()); // A comment has changed position, use the key to determine if we are // displaying this comment and if so move it. Comment movedComment = dataSnapshot.getValue(Comment.class); String commentKey = dataSnapshot.getKey(); // ... } @Override public void onCancelled(DatabaseError databaseError) { Log.w(TAG, "postComments:onCancelled", databaseError.toException()); Toast.makeText(mContext, "Failed to load comments.", Toast.LENGTH_SHORT).show(); } }; databaseReference.addChildEventListener(childEventListener);
Theo dõi các sự kiện giá trị
Mặc dù dùng ChildEventListener
là cách nên dùng để đọc danh sách dữ liệu, nhưng trong một số trường hợp, việc đính kèm ValueEventListener
vào một tài liệu tham khảo danh sách sẽ rất hữu ích.
Việc đính kèm một ValueEventListener
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 DataSnapshot
duy nhất. Sau đó, bạn có thể lặp lại danh sách này để truy cập vào từng thành phần con.
Ngay cả khi chỉ có một kết quả trùng khớp cho truy vấn, ảnh chụp nhanh vẫn là một danh sách; danh sách này chỉ chứa một mục. Để truy cập vào mục này, bạn cần lặp lại kết quả:
Kotlin
// My top posts by number of stars myTopPostsQuery.addValueEventListener(object : ValueEventListener { override fun onDataChange(dataSnapshot: DataSnapshot) { for (postSnapshot in dataSnapshot.children) { // TODO: handle the post } } override fun onCancelled(databaseError: DatabaseError) { // Getting Post failed, log a message Log.w(TAG, "loadPost:onCancelled", databaseError.toException()) // ... } })
Java
// My top posts by number of stars myTopPostsQuery.addValueEventListener(new ValueEventListener() { @Override public void onDataChange(@NonNull DataSnapshot dataSnapshot) { for (DataSnapshot postSnapshot: dataSnapshot.getChildren()) { // TODO: handle the post } } @Override public void onCancelled(@NonNull DatabaseError databaseError) { // Getting Post failed, log a message Log.w(TAG, "loadPost:onCancelled", databaseError.toException()); // ... } });
Mẫu này có thể hữu ích khi bạn muốn tìm nạp tất cả các thành phần con của một danh sách trong một thao tác duy nhất, thay vì theo dõi các sự kiện onChildAdded
bổ sung.
Tách trình nghe
Bạn có thể xoá lệnh gọi lại bằng cách gọi phương thức removeEventListener()
trên tham chiếu cơ sở dữ liệu Firebase.
Nếu một trình nghe đã được thêm nhiều lần vào một vị trí dữ liệu, thì trình nghe đó sẽ được gọi nhiều lần cho mỗi sự kiện và bạn phải tách trình nghe đó cùng số lần để xoá hoàn toàn.
Việc gọi removeEventListener()
trên một trình nghe chính không tự động xoá các trình nghe đã đăng ký trên các nút con của trình nghe đó; bạn cũng phải gọi removeEventListener()
trên mọi trình nghe con để xoá lệnh gọi lại.
Sắp xếp và lọc dữ liệu
Bạn có thể dùng lớp Realtime Database Query
để truy xuất dữ liệu được sắp xếp theo khoá, theo giá trị hoặc theo giá trị của một phần tử con. Bạn cũng có thể lọc kết quả đã sắp xếp theo một số lượng kết quả cụ thể hoặc một dải khoá 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 thức sắp xếp theo để xác định cách sắp xếp kết quả:
Phương thức | Cách sử dụng |
---|---|
orderByChild() |
Sắp xếp kết quả theo giá trị của một khoá con hoặc đường dẫn con lồng nhau được chỉ định. |
orderByKey()
| Sắp xếp kết quả theo khoá con. |
orderByValue() |
Sắp xếp kết quả theo giá trị con. |
Mỗi lần, bạn chỉ có thể sử dụng một phương thức sắp xếp theo. Việc gọi phương thức sắp xếp theo nhiều lần trong cùng một truy vấn sẽ gây ra lỗi.
Ví dụ sau đây minh hoạ 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ố lượng sao:
Kotlin
// My top posts by number of stars val myUserId = uid val myTopPostsQuery = databaseReference.child("user-posts").child(myUserId) .orderByChild("starCount") myTopPostsQuery.addChildEventListener(object : ChildEventListener { // TODO: implement the ChildEventListener methods as documented above // ... })
Java
// My top posts by number of stars String myUserId = getUid(); Query myTopPostsQuery = databaseReference.child("user-posts").child(myUserId) .orderByChild("starCount"); myTopPostsQuery.addChildEventListener(new ChildEventListener() { // TODO: implement the ChildEventListener methods as documented above // ... });
Thao tác này xác định một truy vấn khi kết hợp với một trình nghe con sẽ đồng bộ hoá ứng dụng với 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 mã nhận dạng người dùng của họ, được sắp xếp theo số lượng sao mà mỗi bài đăng nhận được. Kỹ thuật sử dụng mã nhận dạng làm khoá chỉ mục này được gọi là phân phối dữ liệu. Bạn có thể đọc thêm về kỹ thuật này trong phần Cấu trúc cơ sở dữ liệu.
Lệnh gọi đến phương thức orderByChild()
chỉ định khoá con để sắp xếp kết quả theo. Trong trường hợp này, các bài đăng được sắp xếp theo giá trị của thành phần con "starCount"
tương ứng. Bạn cũng có thể sắp xếp các truy vấn theo các phần tử con lồng nhau, trong trường hợp bạn có dữ liệu 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 ví dụ này, chúng ta có thể sắp xếp các phần tử trong danh sách theo các giá trị được lồng trong khoá metrics
bằng cách chỉ định đường dẫn tương đối đến phần tử con được lồng trong lệnh gọi orderByChild()
.
Kotlin
// Most viewed posts val myMostViewedPostsQuery = databaseReference.child("posts") .orderByChild("metrics/views") myMostViewedPostsQuery.addChildEventListener(object : ChildEventListener { // TODO: implement the ChildEventListener methods as documented above // ... })
Java
// Most viewed posts Query myMostViewedPostsQuery = databaseReference.child("posts") .orderByChild("metrics/views"); myMostViewedPostsQuery.addChildEventListener(new ChildEventListener() { // TODO: implement the ChildEventListener methods as documented above // ... });
Để 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 bài viết 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 thức giới hạn hoặc phạm vi nào với phương thức sắp xếp theo khi tạo truy vấn.
Phương thức | Cách sử dụng |
---|---|
limitToFirst() |
Đặt số lượng tối đa các mục cần trả về từ đầu danh sách kết quả có thứ tự. |
limitToLast() |
Đặt số lượng tối đa các mục cần trả về từ cuối danh sách kết quả có thứ tự. |
startAt() |
Trả về các mục lớn hơn hoặc bằng khoá hoặc giá trị đã chỉ định, tuỳ thuộc vào phương thức sắp xếp đã chọn. |
startAfter() |
Trả về các mục lớn hơn khoá hoặc giá trị đã chỉ định, tuỳ thuộc vào phương thức sắp xếp theo bạn chọn. |
endAt() |
Trả về các mục nhỏ hơn hoặc bằng khoá hoặc giá trị được chỉ định, tuỳ thuộc vào phương thức sắp xếp đã chọn. |
endBefore() |
Trả về các mục nhỏ hơn khoá hoặc giá trị được chỉ định, tuỳ thuộc vào phương thức sắp xếp đã chọn. |
equalTo() |
Trả về các mục bằng với khoá hoặc giá trị đã chỉ định, tuỳ thuộc vào phương thức sắp xếp theo bạn chọn. |
Không giống như các phương thức sắp xếp, bạn có thể kết hợp nhiều hàm giới hạn hoặc hàm phạm vi.
Ví dụ: bạn có thể kết hợp phương thức startAt()
và endAt()
để giới hạn kết quả trong một phạm vi giá trị được chỉ định.
Ngay cả khi chỉ có một kết quả khớp cho truy vấn, ảnh chụp nhanh vẫn là một danh sách; danh sách này chỉ chứa một mục duy nhất. Để truy cập vào mục này, bạn cần lặp lại kết quả:
Kotlin
// My top posts by number of stars myTopPostsQuery.addValueEventListener(object : ValueEventListener { override fun onDataChange(dataSnapshot: DataSnapshot) { for (postSnapshot in dataSnapshot.children) { // TODO: handle the post } } override fun onCancelled(databaseError: DatabaseError) { // Getting Post failed, log a message Log.w(TAG, "loadPost:onCancelled", databaseError.toException()) // ... } })
Java
// My top posts by number of stars myTopPostsQuery.addValueEventListener(new ValueEventListener() { @Override public void onDataChange(@NonNull DataSnapshot dataSnapshot) { for (DataSnapshot postSnapshot: dataSnapshot.getChildren()) { // TODO: handle the post } } @Override public void onCancelled(@NonNull DatabaseError databaseError) { // Getting Post failed, log a message Log.w(TAG, "loadPost:onCancelled", databaseError.toException()); // ... } });
Giới hạn số lượng kết quả
Bạn có thể dùng các phương thức limitToFirst()
và limitToLast()
để đặt số lượng tối đa các phần tử con cần đồng bộ hoá cho một lệnh gọi lại nhất định. Ví dụ: nếu bạn dùng limitToFirst()
để đặt giới hạn là 100, ban đầu bạn sẽ chỉ nhận được tối đa 100 lệnh gọi lại onChildAdded()
. Nếu bạn có ít hơn 100 mục được lưu trữ trong cơ sở dữ liệu Firebase, thì một lệnh gọi lại onChildAdded()
sẽ kích hoạt cho từng mục.
Khi các mục thay đổi, bạn sẽ nhận được các lệnh gọi lại onChildAdded()
cho những mục xuất hiện trong truy vấn và các lệnh gọi lại onChildRemoved()
cho những mục không còn trong truy vấn để tổng số vẫn là 100.
Ví dụ sau đây minh hoạ cách ứng dụng blog mẫu xác định một truy vấn để truy xuất danh sách 100 bài đăng gần đây nhất của tất cả người dùng:
Kotlin
// Last 100 posts, these are automatically the 100 most recent // due to sorting by push() keys. databaseReference.child("posts").limitToFirst(100)
Java
// Last 100 posts, these are automatically the 100 most recent // due to sorting by push() keys Query recentPostsQuery = databaseReference.child("posts") .limitToFirst(100);
Ví dụ này chỉ xác định một truy vấn, để thực sự đồng bộ hoá dữ liệu, truy vấn cần có một trình nghe được đính kèm.
Lọc theo khoá hoặc giá trị
Bạn có thể dùng startAt()
, startAfter()
, endAt()
, endBefore()
và equalTo()
để chọn điểm bắt đầu, điểm kết thúc và điểm tương đương tuỳ ý cho các truy vấn. Điều này có thể hữu ích khi phân trang dữ liệu hoặc tìm các mục có phần tử con có một giá trị cụ thể.
Cách sắp xếp dữ liệu truy vấn
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 sắp xếp theo thứ tự trong lớp Query
.
orderByChild
Khi sử dụng orderByChild()
, dữ liệu chứa khoá con được chỉ định sẽ được sắp xếp như sau:
- Những trẻ có giá trị
null
cho khoá con đã chỉ định sẽ xuất hiện trước. - Tiếp theo là các phần tử con có giá trị
false
cho khoá con đã chỉ định. Nếu nhiều thành phần con có giá trịfalse
, thì các thành phần con đó sẽ được sắp xếp theo thứ tự từ điển theo khoá. - Tiếp theo là các phần tử con có giá trị
true
cho khoá con đã chỉ định. Nếu nhiều thành phần con có giá trị làtrue
, thì các thành phần con đó sẽ được sắp xếp theo thứ tự từ điển theo khoá. - Tiếp theo là các phần tử con có giá trị bằng số, được sắp xếp theo thứ tự tăng dần. Nếu nhiều phần tử con có cùng giá trị số cho nút con được chỉ định, thì chúng sẽ được sắp xếp theo khoá.
- Các chuỗi xuất hiện sau các số và được sắp xếp theo thứ tự từ điển tăng dần. Nếu nhiều phần tử con có cùng giá trị cho nút con được chỉ định, thì các phần tử con đó sẽ được sắp xếp theo thứ tự từ điển theo khoá.
- Các đối tượng xuất hiện sau cùng và được sắp xếp theo thứ tự từ điển theo khoá theo thứ tự tăng dần.
orderByKey
Khi bạn dùng orderByKey()
để sắp xếp dữ liệu, dữ liệu sẽ được trả về theo thứ tự tăng dần theo khoá.
- Trẻ em có khoá có thể phân tích cú pháp dưới dạng số nguyên 32 bit sẽ xuất hiện trước, được sắp xếp theo thứ tự tăng dần.
- Tiếp theo là các phần tử con có giá trị chuỗi làm khoá, được sắp xếp theo thứ tự từ điển tăng dần.
orderByValue
Khi sử dụng orderByValue()
, các thành phần con được sắp xếp theo giá trị của chúng. Tiêu chí sắp xếp giống như trong orderByChild()
, ngoại trừ việc giá trị của nút được dùng thay vì giá trị của một khoá con được chỉ định.