الحصول على مرجع قاعدة بيانات
لقراءة البيانات أو كتابتها من قاعدة البيانات، تحتاج إلى مثيل من
DatabaseReference
:
DatabaseReference ref = FirebaseDatabase.instance.ref();
قوائم القراءة والكتابة
إلحاق بقائمة من البيانات
استخدِم الطريقة push()
لإلحاق البيانات بقائمة في تطبيقات متعددة المستخدمين.
تُنشئ طريقة push()
مفتاحًا فريدًا في كل مرة تتم فيها إضافة عنصر فرعي جديد إلى
مرجع Firebase المحدّد. باستخدام مفاتيح التشفير هذه التي يتم إنشاؤها تلقائيًا لكل عنصر
جديد في القائمة، يمكن لعدة عملاء إضافة عناصر فرعية إلى الموقع الجغرافي نفسه
في الوقت نفسه بدون حدوث تعارضات في الكتابة. يستند المفتاح الفريد الذي يتم إنشاؤه من خلال push()
إلى طابع زمني، لذا يتم ترتيب عناصر القائمة تلقائيًا
حسب التسلسل الزمني.
يمكنك استخدام إشارة إلى البيانات الجديدة التي تعرضها الطريقة push()
للحصول على
قيمة مفتاح الطفل الذي تم إنشاؤه تلقائيًا أو ضبط بيانات للطفل. يحتوي الحقل
.key
في مرجع push()
على المفتاح الذي تم إنشاؤه تلقائيًا.
يمكنك استخدام هذه المفاتيح التي يتم إنشاؤها تلقائيًا لتبسيط تسطيح بنية البيانات. للمزيد من المعلومات، يُرجى الاطّلاع على مثال توزيع البيانات.
على سبيل المثال، يمكن استخدام push()
لإضافة مشاركة جديدة إلى قائمة المشاركات
في تطبيق اجتماعي:
DatabaseReference postListRef = FirebaseDatabase.instance.ref("posts");
DatabaseReference newPostRef = postListRef.push();
newPostRef.set({
// ...
});
الاستماع إلى أحداث الطفل
يتم بدء أحداث العناصر الفرعية استجابةً لعمليات معيّنة تحدث ل
العناصر الفرعية لعقدة من عملية، مثل إضافة عنصر فرعي جديد من خلال الأسلوب
push()
أو تعديل عنصر فرعي من خلال الأسلوب update()
.
الحدث | الاستخدام المعتاد |
---|---|
onChildAdded |
استرداد قوائم العناصر أو الاستماع إلى الإضافات إلى قائمة من العناصر. يتم بدء هذا الحدث مرة واحدة لكل عنصر فرعي حالي، ثم مرة أخرى في كل مرة تتم فيها إضافة عنصر فرعي جديد إلى المسار المحدّد. يتم إرسال لقطة إلى المستمع تحتوي على بيانات الطفل الجديد. |
onChildChanged |
رصد التغييرات على العناصر في القائمة يتم تشغيل هذا الحدث في أي وقت يتم فيه تعديل عقدة فرعية. ويشمل ذلك أي تعديلات على العناصر الفرعية للعقدة الفرعية. تحتوي اللقطة التي تم تمريرها إلى مستمع الحدث على البيانات المعدَّلة للطفل. |
onChildRemoved |
رصد العناصر التي تتم إزالتها من القائمة يتم بدء هذا الحدث عند إزالة عنصر فرعي مباشر. تحتوي اللقطة التي تم تمريرها إلى كتلة الاستدعاء على بيانات العنصر الفرعي الذي تمّت إزالته. |
onChildMoved |
استمع إلى التغييرات في ترتيب العناصر في قائمة مرتبة. تتبع أحداث onChildMoved دائمًا حدث onChildChanged الذي تسبّب في تغيير ترتيب العنصر (استنادًا إلى طريقة الترتيب الحالية). |
يمكن أن يكون كلّ من هذه الأدوات مفيدًا معًا للاستماع إلى التغييرات التي تطرأ على node معيّنة في قاعدة بيانات. على سبيل المثال، قد يستخدم تطبيق التدوين الاجتماعي هذه الأساليب معًا لمراقبة النشاط في تعليقات المشاركة، كما هو موضح أدناه:
final commentsRef = FirebaseDatabase.instance.ref("post-comments/$postId");
commentsRef.onChildAdded.listen((event) {
// A new comment has been added, so add it to the displayed list.
});
commentsRef.onChildChanged.listen((event) {
// A comment has changed; use the key to determine if we are displaying this
// comment and if so displayed the changed comment.
});
commentsRef.onChildRemoved.listen((event) {
// A comment has been removed; use the key to determine if we are displaying
// this comment and if so remove it.
});
رصد أحداث القيم
على الرغم من أنّ رصد الأحداث الفرعية هو الطريقة المُقترَحة لقراءة قوائم البيانات، فهناك حالات يكون فيها رصد أحداث القيمة في مرجع قائمة مفيدًا.
سيؤدي إرفاق مستمع value
بقائمة بيانات إلى عرض
القائمة الكاملة للبيانات كنبذة واحدة يمكنك بعد ذلك تكرارها
للوصول إلى الأطفال الفرديين.
حتى في حال توفّر مطابقة واحدة فقط لطلب البحث، تظلّ اللقطة قائمة، ولكنها تحتوي على عنصر واحد فقط. للوصول إلى العنصر، عليك تكرار قراءة النتيجة:
myTopPostsQuery.onValue.listen((event) {
for (final child in event.snapshot.children) {
// Handle the post.
}
}, onError: (error) {
// Error.
});
يمكن أن يكون هذا النمط مفيدًا عندما تريد جلب جميع العناصر الفرعية لقائمة في عملية واحدة، بدلاً من الاستماع إلى أحداث إضافية لإضافة عنصر فرعي .
فرز البيانات وتصفيتها
يمكنك استخدام فئة Query
لاسترداد البيانات مفروضة حسب
المفتاح أو القيمة أو قيمة العنصر الفرعي. يمكنك أيضًا فلترة
النتيجة المرتبطة بعدد معيّن من النتائج أو نطاق من المفاتيح أو
القيم.
ترتيب البيانات
لاسترداد البيانات المفروَضة، ابدأ بتحديد إحدى طرق "الترتيب حسب" لتحديد كيفية ترتيب النتائج:
الطريقة | الاستخدام |
---|---|
orderByChild() |
ترتيب النتائج حسب قيمة مفتاح فرعي محدّد أو مسار فرعي متداخل |
orderByKey() |
ترتيب النتائج حسب مفاتيح العناصر الفرعية |
orderByValue() |
ترتيب النتائج حسب قيم السمات الثانوية |
يمكنك استخدام طريقة واحدة فقط لترتيب النتائج في كل مرة. يؤدي استدعاء طريقة ترتيب بالاستناد إلى عمود معيّن مرارًا وتكرارًا في طلب البحث نفسه إلى ظهور خطأ.
يوضّح المثال التالي كيفية استرداد قائمة بأحد مستخدمي Flickr أهم مشاركاته مرتبة حسب عدد النجوم:
final myUserId = FirebaseAuth.instance.currentUser?.uid;
final topUserPostsRef = FirebaseDatabase.instance
.ref("user-posts/$myUserId")
.orderByChild("starCount");
يحدد هذا استعلامًا عند دمجه مع مستمع ثانوي ويؤدي إلى مزامنة العميل مع مشاركات المستخدم من المسار في قاعدة البيانات استنادًا إلى معرّف المستخدم، ويتم ترتيبها حسب عدد النجوم التي تلقتها كل مشاركة. تُعرف تقنية استخدام المعرّفات كمفاتيح الفهرس باسم "نشر البيانات"، ويمكنك الاطّلاع علىمزيد من المعلومات حولها في مقالة تنظيم قاعدة بياناتك.
تحدِّد الدعوة إلى الطريقة orderByChild()
مفتاح الطفل لترتيب
النتائج. في هذه الحالة، يتم ترتيب المشاركات حسب قيمة "starCount"
التابعة لها. يمكن أيضًا ترتيب طلبات البحث حسب العناصر المُدمجة
التابعة، في حال كانت لديك بيانات بالشكل التالي:
"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",
}
},
في هذه الحالة، يمكننا ترتيب عناصر القائمة حسب القيم المتداخلة ضمن المفتاح metrics
من خلال تحديد المسار النسبي للعنصر الفرعي المدمج في طلب orderByChild()
.
final mostViewedPosts =
FirebaseDatabase.instance.ref('posts').orderByChild('metrics/views');
لمزيد من المعلومات عن كيفية ترتيب أنواع البيانات الأخرى، اطّلِع على كيفية ترتيب بيانات طلبات البحث.
تصفية البيانات
لفلترة البيانات، يمكنك دمج أيّ من طرق الحدّ أو النطاق مع طريقة order-by عند إنشاء طلب بحث.
الطريقة | الاستخدام |
---|---|
limitToFirst() |
تُستخدَم لتحديد الحد الأقصى لعدد العناصر التي سيتم عرضها من بداية القائمة المرتبة للنتائج. |
limitToLast() |
تتيح هذه السياسة ضبط الحد الأقصى لعدد العناصر المطلوب عرضها من نهاية قائمة النتائج المرتبة. |
startAt() |
عرض العناصر التي تكون أكبر من أو تساوي المفتاح أو القيمة المحدّدة، استنادًا إلى طريقة الترتيب المحدّدة |
startAfter() |
عرض العناصر التي تزيد عن المفتاح أو القيمة المحدّدة استنادًا إلى طريقة الترتيب التي تم اختيارها |
endAt() |
يمكنك عرض عناصر أقل من أو تساوي المفتاح أو القيمة المحدّدة، اعتمادًا على طريقة الترتيب المحدّدة. |
endBefore() |
عرض العناصر التي تقلّ عن المفتاح أو القيمة المحدّدة استنادًا إلى طريقة الترتيب المحدّدة |
equalTo() |
عرض العناصر التي تساوي المفتاح أو القيمة المحدّدة، استنادًا إلى طريقة الترتيب التي تم اختيارها |
على عكس طرق "الترتيب حسب"، يمكنك دمج عدّة دوالّ حدّ أو نطاق.
على سبيل المثال، يمكنك دمج الطريقتَين startAt()
وendAt()
للحدّ من
النتائج إلى نطاق محدّد من القيم.
الحد من عدد النتائج
يمكنك استخدام الطريقتين limitToFirst()
وlimitToLast()
لضبط حد أقصى لعدد العناصر الثانوية المطلوب مزامنتها لحدث معيّن. على سبيل المثال، إذا
استخدمت limitToFirst()
لضبط الحدّ الأقصى البالغ 100 حدث، ستتلقّى في البداية ما يصل إلى 100 حدث onChildAdded
فقط. إذا كان لديك أقل من 100 عنصر مخزّن في قاعدة بيانات
Firebase، يتمّ بدء حدث onChildAdded
لكلّ عنصر.
عندما تتغيّر العناصر، تتلقّى onChildAdded
حدثًا للعناصر التي تدخل
الطلب وonChildRemoved
حدثًا للعناصر التي تخرج منه كي يظل
إجمالي العدد 100.
يوضّح المثال التالي كيفية تحديد تطبيق التدوين مثالًا طلب بحث ل retrieving list of the 100 most recent posts by all users:
final recentPostsRef = FirebaseDatabase.instance.ref('posts').limitToLast(100);
يحدِّد هذا المثال طلب بحث فقط، ولكن لمزامنة البيانات فعليًا، يجب أن يكون لديه مستمع مرفق.
الفلترة حسب المفتاح أو القيمة
يمكنك استخدام startAt()
وstartAfter()
وendAt()
وendBefore()
وequalTo()
لاختيار نقاط بداية ونهاية وتكافؤ عشوائية
لطلبات البحث. يمكن أن يكون ذلك مفيدًا لتقسيم البيانات إلى صفحات أو العثور على عناصر لها عناصر فرعية
تتضمن قيمة معيّنة.
كيف يتم ترتيب بيانات طلبات البحث
يشرح هذا القسم كيفية ترتيب البيانات حسب كل طريقة من الطرق بالترتيب في
الفئة Query
.
orderByChild
عند استخدام orderByChild()
، يتم ترتيب البيانات التي تحتوي على المفتاح الفرعي المحدّد على النحو التالي:
- تظهر أولاً العناصر الفرعية التي تحتوي على قيمة
null
لمفتاح العنصر الفرعي المحدّد. - تأتي بعد ذلك القيم التي تحتوي على
false
لمفتاح الطفل المحدّد. إذا كانت قيمة عناصر متعددة هيfalse
، يتم ترتيبها ألفبائيًا حسب المفتاح. - تأتي بعد ذلك القيم التي تحتوي على
true
لمفتاح الطفل المحدّد. إذا كانت قيمة عناصر فرعية متعددة هيtrue
، يتم ترتيبها أبجديًا حسب المفتاح. - تأتي الأطفال ذوي القيمة الرقمية بعد ذلك، مرتبة بترتيب تصاعدي. إذا كانت عدّة عناصر فرعية لها القيمة الرقمية نفسها لعنصر فرعي محدّد، يتم ترتيبها حسب المفتاح.
- تأتي السلاسل بعد الأرقام ويتم ترتيبها بشكل تسلسلي تصاعديًا. إذا كانت عدّة عناصر فرعية لها القيمة نفسها لعنصر فرعي محدّد، يتم ترتيبها أبجديًا حسب المفتاح.
- تأتي الكائنات في النهاية ويتم فرزها قاموسًا حسب المفتاح بترتيب تصاعدي.
orderByKey
عند استخدام orderByKey()
لترتيب بياناتك، يتم عرض البيانات بترتيب تصاعدي حسب المفتاح.
- تظهر أولاً العناصر الفرعية التي تحتوي على مفتاح يمكن تحليله كعدد صحيح 32 بت، ويتم ترتيبها تصاعديًا.
- يأتي العناصر الثانوية التي لها قيمة سلسلة كمفتاحها بعد ذلك، ويتم فرزها ترتيبًا تصاعديًا.
orderByValue
عند استخدام orderByValue()
، يتم ترتيب العناصر الثانوية حسب قيمتها. تكون معايير الترتيب
نفسها كما في orderByChild()
، باستثناء أنّه يتم استخدام قيمة العقدة بدلاً من قيمة مفتاح فرعي محدّد.
فصل المستمعين
تتم إزالة طلبات إعادة الاتصال من خلال استدعاء طريقة off()
في
مرجع قاعدة بيانات Firebase.
يمكنك إزالة مستمع واحد من خلال تمريره كمَعلمة إلى off()
.
يؤدي استدعاء off()
على الموقع الجغرافي بدون وسيطات إلى إزالة جميع المستمعين في ذلك
الموقع الجغرافي.
لا يؤدي استدعاء off()
في مستمع رئيسي إلى
إزالة المستمعين المسجَّلين في العقد الفرعية تلقائيًا،
ويجب أيضًا استدعاء off()
في أي مستمعين فرعيين
لإزالة طلب الاستدعاء.