Lấy tham chiếu cơ sở dữ liệu
Để đọc hoặc ghi dữ liệu từ cơ sở dữ liệu, bạn cần một thực thể của firebase.database.Reference:
Web
import { getDatabase } from "firebase/database"; const database = getDatabase();
Web
var database = firebase.database();
Đọ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() 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, nhiều ứng dụ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 bị xung đột khi 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ẽ đượ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 push() trả về để lấy giá trị của khoá được tạo tự động của phần tử con hoặc đặt dữ liệu cho phần tử con. Thuộc tính .key của tham chiếu push() chứ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 đầu ra dữ liệu .
Ví dụ: bạn có thể sử dụng push() để thêm bài đăng mới vào danh sách bài đăng trong một ứng dụng mạng xã hội:
Web
import { getDatabase, ref, push, set } from "firebase/database"; // Create a new post reference with an auto-generated id const db = getDatabase(); const postListRef = ref(db, 'posts'); const newPostRef = push(postListRef); set(newPostRef, { // ... });
Web
// Create a new post reference with an auto-generated id var postListRef = firebase.database().ref('posts'); var newPostRef = postListRef.push(); newPostRef.set({ // ... });
Theo dõi sự kiện phần tử con
Các sự kiện phần tử con được kích hoạt để phản hồi các thao tác cụ thể xảy ra với các phần tử con của một nút từ một thao tác, chẳng hạn như một phần tử con mới được thêm thông qua phương thức push() hoặc một phần tử con đang được cập nhật thông qua phương thức update().
| Sự kiện | Cách sử dụng thông thường |
|---|---|
child_added |
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. Sự kiện này được kích hoạt một lần cho mỗi phần tử con hiện có, sau đó 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. Trình nghe được truyền một ảnh chụp nhanh chứa dữ liệu của phần tử con mới. |
child_changed |
Theo dõi các 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 một nút phần tử con được sửa đổi. Điều này bao gồm mọi sửa đổi đối với các phần tử con của nút phần tử con. Ảnh chụp nhanh được truyền đến trình nghe sự kiện chứa dữ liệu đã cập nhật cho phần tử con. |
child_removed |
Theo dõi các mục bị xoá khỏi danh sách. Sự kiện này được kích hoạt khi một phần tử con trực tiếp bị xoá.Ảnh chụp nhanh được truyền đến khối lệnh gọi lại chứa dữ liệu cho phần tử con bị xoá. |
child_moved |
Theo dõi các thay đổi đối với thứ tự của các mục trong danh sách được sắp xếp.
child_moved sự kiện luôn tuân theo sự kiện child_changed khiến thứ tự của mục thay đổi (dựa trên phương thức sắp xếp hiện tại).
|
Mỗi sự kiện này có thể hữu ích cho việc theo dõi 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 blog xã hội có thể sử dụng các phương thức này cùng nhau để theo dõi hoạt động trong phần bình luận của một bài đăng, như minh hoạ bên dưới:
Web
import { getDatabase, ref, onChildAdded, onChildChanged, onChildRemoved } from "firebase/database"; const db = getDatabase(); const commentsRef = ref(db, 'post-comments/' + postId); onChildAdded(commentsRef, (data) => { addCommentElement(postElement, data.key, data.val().text, data.val().author); }); onChildChanged(commentsRef, (data) => { setCommentValues(postElement, data.key, data.val().text, data.val().author); }); onChildRemoved(commentsRef, (data) => { deleteComment(postElement, data.key); });
Web
var commentsRef = firebase.database().ref('post-comments/' + postId); commentsRef.on('child_added', (data) => { addCommentElement(postElement, data.key, data.val().text, data.val().author); }); commentsRef.on('child_changed', (data) => { setCommentValues(postElement, data.key, data.val().text, data.val().author); }); commentsRef.on('child_removed', (data) => { deleteComment(postElement, data.key); });
Theo dõi sự kiện giá trị
Mặc dù theo dõi sự kiện phần tử con là cách được đề xuất để đọc danh sách dữ liệu, nhưng trong một số trường hợp, việc theo dõi sự kiện giá trị trên tham chiếu danh sách sẽ hữu ích.
Việc đính kèm trình quan sát value 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 duy nhất mà sau đó bạn có thể lặp lại để truy cập vào từng phần tử con.
Ngay cả khi chỉ có một kết quả phù hợp cho truy vấn, ảnh chụp nhanh vẫn là một danh sách; ảnh chụp nhanh chỉ chứa một mục. Để truy cập vào mục đó, bạn cần lặp lại kết quả:
Web
import { getDatabase, ref, onValue } from "firebase/database"; const db = getDatabase(); const dbRef = ref(db, '/a/b/c'); onValue(dbRef, (snapshot) => { snapshot.forEach((childSnapshot) => { const childKey = childSnapshot.key; const childData = childSnapshot.val(); // ... }); }, { onlyOnce: true });
Web
ref.once('value', (snapshot) => { snapshot.forEach((childSnapshot) => { var childKey = childSnapshot.key; var childData = childSnapshot.val(); // ... }); });
Mẫu này có thể hữu ích khi bạn muốn tìm nạp tất cả các phần tử 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 phần tử con được thêm khác.
Sắp xếp và lọc dữ liệu
Bạn có thể sử 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 thành một số lượng kết quả cụ thể hoặc một phạm vi 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 để 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á phần tử con được chỉ định hoặc đường dẫn phần tử con lồng nhau. |
orderByKey()
| Sắp xếp kết quả theo khoá phần tử con. |
orderByValue() |
Sắp xếp kết quả theo giá trị phần tử con. |
Mỗi lần, bạn chỉ có thể sử dụng một phương thức sắp xếp. Việc gọi phương thức sắp xếp 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ượt thích:
Web
import { getDatabase, ref, query, orderByChild } from "firebase/database"; import { getAuth } from "firebase/auth"; const db = getDatabase(); const auth = getAuth(); const myUserId = auth.currentUser.uid; const topUserPostsRef = query(ref(db, 'user-posts/' + myUserId), orderByChild('starCount'));
Web
var myUserId = firebase.auth().currentUser.uid; var topUserPostsRef = firebase.database().ref('user-posts/' + myUserId).orderByChild('starCount');
Điều này xác định một truy vấn mà khi kết hợp với trình nghe phần tử 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ã người dùng của họ, được sắp xếp theo số lượt thích mà mỗi bài đăng nhận được. Kỹ thuật sử dụng mã làm khoá chỉ mục này được gọi là phân đầu ra 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.
Lệnh gọi đến phương thức orderByChild() chỉ định khoá phần tử 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 phần tử con
tương ứng "starCount". 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 trường hợp 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 bên dưới khoá metrics bằng cách chỉ định đường dẫn tương đối đến phần tử con lồng nhau trong lệnh gọi orderByChild().
Web
import { getDatabase, ref, query, orderByChild } from "firebase/database"; const db = getDatabase(); const mostViewedPosts = query(ref(db, 'posts'), orderByChild('metrics/views'));
Web
var mostViewedPosts = firebase.database().ref('posts').orderByChild('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 thức giới hạn hoặc phạm vi nào với phương thức sắp xếp khi tạo truy vấn.
| Phương thức | Cách sử dụng |
|---|---|
limitToFirst() |
Đặt số lượng mục tối đa cần trả về từ đầu danh sách kết quả được sắp xếp. |
limitToLast() |
Đặt số lượng mục tối đa cần trả về từ cuối danh sách kết quả được sắp xếp. |
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 được 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 được chọn. |
endAt() |
Trả về các mục nhỏ 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 được chọn. |
endBefore() |
Trả về các mục nhỏ hơn khoá hoặc giá trị đã chỉ định tuỳ thuộc vào phương thức sắp xếp được chọn. |
equalTo() |
Trả về các mục bằng khoá hoặc giá trị đã chỉ định, tuỳ thuộc vào phương thức sắp xếp được 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 phạm vi.
Ví dụ: bạn có thể kết hợp các 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.
Giới hạn số lượng kết quả
Bạn có thể sử dụng các phương thức limitToFirst() và limitToLast() để đặt số lượng phần tử con tối đa cần đồng bộ hoá cho một sự kiện nhất định. Ví dụ: nếu bạn sử dụng limitToFirst() để đặt giới hạn là 100, thì ban đầu bạn chỉ nhận được tối đa 100 sự kiện child_added. Nếu bạn có ít hơn 100 mục được lưu trữ trong cơ sở dữ liệu Firebase, thì sự kiện child_added sẽ kích hoạt cho mỗi mục.
Khi các mục thay đổi, bạn sẽ nhận được các sự kiện child_added cho các mục nhập truy vấn và các sự kiện child_removed cho các mục bị loại bỏ khỏi 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:
Web
import { getDatabase, ref, query, limitToLast } from "firebase/database"; const db = getDatabase(); const recentPostsRef = query(ref(db, 'posts'), limitToLast(100));
Web
var recentPostsRef = firebase.database().ref('posts').limitToLast(100);
Lọc theo khoá hoặc giá trị
Bạn có thể sử dụng startAt(), startAfter(),endAt(), endBefore() và equalTo() để chọn các điểm bắt đầu, kết thúc và tương đương tuỳ ý cho các truy vấn. Điều này có thể hữu ích cho việc phân trang dữ liệu hoặc tìm các mục có phần tử con có 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 sắp xếp dữ liệu theo từng phương thức sắp xếp trong lớp Query.
orderByChild
Khi sử dụng orderByChild(), dữ liệu chứa khoá phần tử con được chỉ định sẽ được sắp xếp như sau:
- Các phần tử con có giá trị
nullcho khoá phần tử con được chỉ định sẽ xuất hiện đầu tiên. - Các phần tử con có giá trị
falsecho khoá phần tử con được chỉ định 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 thứ tự từ điển theo khoá. - Các phần tử con có giá trị
truecho khoá phần tử con được chỉ định xuất hiện tiếp theo. Nếu nhiều phần tử con có giá trịtrue, chúng sẽ được sắp xếp theo thứ tự từ điển theo khoá. - Các phần tử con có giá trị số sẽ xuất hiện tiếp theo, đượ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 phần tử con được chỉ định, 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 phần tử con được chỉ định, chúng sẽ được sắp xếp theo thứ tự từ điển theo khoá.
- Các đối tượng xuất hiện cuối 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 sử 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á.
- Các phần tử con có khoá có thể được phân tích cú pháp dưới dạng số nguyên 32 bit sẽ xuất hiện đầu tiên, được sắp xếp theo thứ tự tăng dần.
- Các phần tử con có giá trị chuỗi làm khoá sẽ xuất hiện tiếp theo, được sắp xếp theo thứ tự từ điển tăng dần.
orderByValue
Khi sử dụng orderByValue(), các phần tử con sẽ đượ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 sử dụng thay vì giá trị của khoá phần tử con được chỉ định.
Tách trình nghe
Các lệnh gọi lại sẽ bị xoá bằng cách gọi phương thức off() trên tham chiếu cơ sở dữ liệu Firebase.
Bạn có thể xoá một trình nghe bằng cách truyền trình nghe đó làm tham số cho off().
Việc gọi off() trên vị trí mà không có đối số sẽ xoá tất cả các trình nghe tại vị trí đó.
Việc gọi off() trên trình nghe mẹ không tự động xoá các trình nghe đã đăng ký trên các nút phần tử con của trình nghe đó; bạn cũng phải gọi off() trên mọi trình nghe phần tử con để xoá lệnh gọi lại.