پایگاه داده خود را ساختار دهید

این راهنما برخی از مفاهیم کلیدی در معماری داده و بهترین شیوه ها برای ساختاردهی داده های JSON در Firebase Realtime Database شما را پوشش می دهد.

ساختن یک پایگاه داده با ساختار مناسب نیاز به کمی پیش بینی دارد. مهمتر از همه، شما باید برای چگونگی ذخیره و بازیابی داده ها برنامه ریزی کنید تا آن فرآیند تا حد امکان آسان شود.

نحوه ساختار داده ها: این یک درخت JSON است

همه داده های Firebase Realtime Database به عنوان اشیاء JSON ذخیره می شوند. می توانید پایگاه داده را به عنوان درخت JSON میزبان ابری در نظر بگیرید. برخلاف پایگاه داده SQL، هیچ جدول یا رکوردی وجود ندارد. هنگامی که داده ها را به درخت JSON اضافه می کنید، به یک گره در ساختار JSON موجود با یک کلید مرتبط تبدیل می شود. می‌توانید کلیدهای خود را مانند شناسه‌های کاربری یا نام‌های معنایی ارائه کنید، یا می‌توانید با استفاده از childByAutoId برای شما ارائه شوند.

اگر کلیدهای خود را ایجاد می‌کنید، باید دارای کد UTF-8 باشند، حداکثر می‌توانند 768 بایت باشند و نمی‌توانند حاوی . , $ , # , [ , ] , / یا نویسه های کنترل ASCII 0-31 یا 127. شما نمی توانید از نویسه های کنترل ASCII در خود مقادیر نیز استفاده کنید.

به عنوان مثال، یک برنامه چت را در نظر بگیرید که به کاربران اجازه می دهد پروفایل اصلی و لیست مخاطبین را ذخیره کنند. یک نمایه کاربر معمولی در مسیری مانند /users/$uid قرار دارد. کاربر alovelace ممکن است ورودی پایگاه داده ای داشته باشد که چیزی شبیه به این است:

{
  "users": {
    "alovelace": {
      "name": "Ada Lovelace",
      "contacts": { "ghopper": true },
    },
    "ghopper": { ... },
    "eclarke": { ... }
  }
}

اگرچه پایگاه داده از درخت JSON استفاده می کند، داده های ذخیره شده در پایگاه داده را می توان به عنوان انواع بومی خاصی که با انواع JSON موجود مطابقت دارند نشان داد تا به شما در نوشتن کد قابل نگهداری بیشتر کمک کند.

بهترین روش ها برای ساختار داده

از تودرتو کردن داده ها خودداری کنید

از آنجایی که Firebase Realtime Database اجازه می دهد تا داده های تودرتو تا عمق 32 سطح را در خود جای دهد، ممکن است وسوسه شوید که این ساختار پیش فرض باشد. با این حال، هنگامی که داده ها را در مکانی در پایگاه داده خود واکشی می کنید، تمام گره های فرزند آن را نیز بازیابی می کنید. علاوه بر این، هنگامی که به کسی اجازه می‌دهید در یک گره در پایگاه داده خود دسترسی خواندن یا نوشتن داشته باشد، به او اجازه دسترسی به تمام داده‌های تحت آن گره را نیز می‌دهید. بنابراین، در عمل، بهتر است ساختار داده خود را تا حد امکان صاف نگه دارید.

برای مثالی از اینکه چرا داده های تودرتو بد هستند، ساختار چند تو در تو زیر را در نظر بگیرید:

{
  // This is a poorly nested data architecture, because iterating the children
  // of the "chats" node to get a list of conversation titles requires
  // potentially downloading hundreds of megabytes of messages
  "chats": {
    "one": {
      "title": "Historical Tech Pioneers",
      "messages": {
        "m1": { "sender": "ghopper", "message": "Relay malfunction found. Cause: moth." },
        "m2": { ... },
        // a very long list of messages
      }
    },
    "two": { ... }
  }
}

با این طراحی تودرتو، تکرار از طریق داده ها مشکل ساز می شود. به عنوان مثال، فهرست کردن عناوین مکالمات چت نیاز به دانلود کل درخت chats ، از جمله همه اعضا و پیام‌ها، برای مشتری دارد.

صاف کردن ساختارهای داده

اگر داده‌ها به جای آن به مسیرهای جداگانه تقسیم شوند، که به آن غیرعادی‌سازی نیز می‌گویند، می‌توان آن‌ها را در فراخوان‌های جداگانه، در صورت نیاز، بارگیری کرد. این ساختار مسطح را در نظر بگیرید:

{
  // Chats contains only meta info about each conversation
  // stored under the chats's unique ID
  "chats": {
    "one": {
      "title": "Historical Tech Pioneers",
      "lastMessage": "ghopper: Relay malfunction found. Cause: moth.",
      "timestamp": 1459361875666
    },
    "two": { ... },
    "three": { ... }
  },

  // Conversation members are easily accessible
  // and stored by chat conversation ID
  "members": {
    // we'll talk about indices like this below
    "one": {
      "ghopper": true,
      "alovelace": true,
      "eclarke": true
    },
    "two": { ... },
    "three": { ... }
  },

  // Messages are separate from data we may want to iterate quickly
  // but still easily paginated and queried, and organized by chat
  // conversation ID
  "messages": {
    "one": {
      "m1": {
        "name": "eclarke",
        "message": "The relay seems to be malfunctioning.",
        "timestamp": 1459361875337
      },
      "m2": { ... },
      "m3": { ... }
    },
    "two": { ... },
    "three": { ... }
  }
}

اکنون می توان با بارگیری چند بایت در هر مکالمه، واکشی سریع ابرداده برای فهرست کردن یا نمایش اتاق ها در یک UI، فهرست اتاق ها را تکرار کرد. پیام‌ها را می‌توان به‌طور جداگانه واکشی کرد و به محض رسیدن نمایش داد، و به رابط کاربری اجازه می‌دهد پاسخگو و سریع بماند.

داده هایی ایجاد کنید که مقیاس شوند

هنگام ساخت اپلیکیشن، اغلب بهتر است زیرمجموعه ای از یک لیست را دانلود کنید. این امر به ویژه در صورتی رایج است که فهرست حاوی هزاران رکورد باشد. هنگامی که این رابطه ثابت و یک جهته است، می توانید به سادگی اشیاء کودک را زیر والد لانه کنید.

گاهی اوقات، این رابطه پویاتر است، یا ممکن است لازم باشد این داده ها غیرعادی شوند. بسیاری از مواقع می‌توانید با استفاده از یک پرس و جو برای بازیابی زیرمجموعه‌ای از داده‌ها، همانطور که در Retrieve Data بحث شد، داده‌ها را غیرعادی کنید.

اما حتی این ممکن است ناکافی باشد. به عنوان مثال، یک رابطه دو طرفه بین کاربران و گروه ها را در نظر بگیرید. کاربران می توانند به یک گروه تعلق داشته باشند و گروه ها لیستی از کاربران را تشکیل می دهند. وقتی زمان تصمیم گیری فرا می رسد که یک کاربر به کدام گروه تعلق دارد، همه چیز پیچیده می شود.

چیزی که لازم است روشی زیبا برای فهرست کردن گروه‌هایی است که یک کاربر به آنها تعلق دارد و فقط داده‌های آن گروه‌ها را واکشی می‌کند. فهرستی از گروه‌ها می‌تواند در اینجا کمک زیادی کند:

// An index to track Ada's memberships
{
  "users": {
    "alovelace": {
      "name": "Ada Lovelace",
      // Index Ada's groups in her profile
      "groups": {
         // the value here doesn't matter, just that the key exists
         "techpioneers": true,
         "womentechmakers": true
      }
    },
    ...
  },
  "groups": {
    "techpioneers": {
      "name": "Historical Tech Pioneers",
      "members": {
        "alovelace": true,
        "ghopper": true,
        "eclarke": true
      }
    },
    ...
  }
}

ممکن است متوجه شوید که با ذخیره کردن رابطه در زیر رکورد Ada و زیر گروه، برخی از داده ها را کپی می کند. اکنون alovelace تحت یک گروه ایندکس شده است و techpioneers در نمایه Ada فهرست شده است. پس برای حذف آدا از گروه باید در دو جا آپدیت شود.

این یک افزونگی ضروری برای روابط دو طرفه است. این به شما امکان می‌دهد تا به سرعت و کارآمد عضویت‌های Ada را دریافت کنید، حتی زمانی که فهرست کاربران یا گروه‌ها به میلیون‌ها نفر می‌رسد یا زمانی که قوانین امنیتی Realtime Database مانع از دسترسی به برخی از سوابق می‌شوند.

این رویکرد، معکوس کردن داده‌ها با فهرست کردن شناسه‌ها به‌عنوان کلید و تنظیم مقدار روی true، بررسی یک کلید را به سادگی خواندن /users/$uid/groups/$group_id و بررسی اینکه آیا آن null است یا خیر، می‌کند. ایندکس سریعتر و کارآمدتر از پرس و جو یا اسکن داده است.

مراحل بعدی