migration با پایگاهداده محلی در اپلیکیشنهای موبایلی همیشه برای من شبیه به خنثی کردن یک بمب بوده. اما کتابخانه room این کار را تاحدی راحتتر کرده است.
پایگاهداده محلی براساس منطق تجاری شخصی(business logic) طراحی شده است، و فقط مسئله زمان است که قبل از اینکه با یک موقعیت مواجه شوید باید با انواع مختلف مهاجرت آشنا شده باشید.
بهجای اینکه دنبال راه حل باشید، سعی کنید یاد بگیرید migration چگونه در کتابخانه room کار میکند. بنابراین میتوانید کار درست را در جای درست انجام دهید.
با room، اگر شمای پایگاهداده را عوض کنید یا جدول جدید اضافه کنید و نسخه پایگاهداده را بروز نکنید، اپلیکیشن شما crash میکند. همچنین اگر نسخه پایگاهداده را بروز کنید و هیچگونه قاعده migrationای برای آن قرار ندهید اپلیکیشن crash خواهد کرد.
درصورتیکه قاعدهی migration مناسب نباشد، جداول پایگاهداده ازبین میرود، و کاربران شما دادههایشان را از دست خواهند داد.
میدانم که پردازش بسیار زیادی است، اما اگر یک بار یادبگیرید که migration چگونه کار میکند، مطمئن هستم که دیگر احساسی نخواهید کرد.
پس بیایید شروع کنیم :)
تاریخچه
Room یک لایهی انتزاعی روی پایگاهداده SQLite است، که یعنی room باید با SQLite توافق کند.
بنابراین، اگر میدانید که migration در SQLite چگونه کار میکند؛ میدانید که چرا نیاز دارید که کار های خاصی مانند بروزرسانی نسخه، قاعده migration، و موارد دیگری که در ادامه خواهید دید در پایگاهداده room انجام دهید.
پایگاهداده SQLite تغییرات شِما را با نسخه پایگاهداده کنترل میکند. برای خاصتر بودن، هر بار که جداول پایگاهداده حذف، اضافه یا شِمای آنها تغییر کرد باید نسخه پایگاهداده اضافه شود.
بنابراین وقتی اپلیکیشن را باز میکنید، Sqlite ابتدا migration نسخه را مدنظر قرار میدهد، و سپس پایگاهداده را باز خواهد کرد.
موارد اجباری که در room باید با آن مقابله کنیم
بیایید که یک مثال ساده را در نظر بگیریم. ما یک پایگاهداده داریم که یک جدول به نام Movies که شامل سه ستون ID(primary key)، name و cover pic که نوع دادههایشان به ترتیب Int, String, String میباشد است.
@Entity(tableName = "MOVIES TABLE")
data class Movies(
@PrimaryKey @ColumnInfo(name = "id")
var id: Int,
@ColumnInfo(name = "name")
var name: String,
@ColumnInfo(name = "coverpic")
var coverPic: String,
) {
constructor() : this("-1", "", "")
}
اکنون به یک ستون جدید در جدول movies نیاز داریم، بنابراین یک متغییر جدید در کلاس movies همانند زیر اضافه میکنیم.
@Entity(tableName = "MOVIES TABLE")
data class Movies(
@PrimaryKey @ColumnInfo(name = "id")
var id: Int,
@ColumnInfo(name = "name")
var name: String,
@ColumnInfo(name = "coverpic")
var coverPic: String,
@ColumnInfo(name = "rating")
var rating: String
) {
constructor() : this("-1", "", "","0")
}
مورد اول نسخه بروزرسانی نشده است
اکنون کار شما با تغییرات پایگاهداده تمام شده است و اپلیکیشن را اجرا میکنید، حالا زمانی که میخواهید به room دسترسی داشته باشید به خطایی همانند زیر برمیخورید.
java.lang.IllegalStateException: Room cannot verify the data integrity.
Looks like you've changed schema but forgot to update the version number.
You can simply fix this by increasing the version number.
اکنون room سعی در شناسایی شناسه پایگاهداده با مقایسه با شناسهی hash شدهی نسخه کنونی، که در room_master_table ذخیره شده است. اما هیچگونه شناسهی hash شدهای وجود ندارد، بنابراین اپلیکیشن با illegalStateException کرش خواهد کرد.
حال شما علت اولین crash را میدانید، پس بیاید این رو داخل یک چک لیست قرار بدیم که در نسخه از اپلیکیشن آن را بررسی کنیم.
ما میتوانیم این مشکل را تنها با بروزرسانی نسخهی پایگاهداده در کلاس پایگاهداده و یا کلاسی که از RoomDatabase() ارث برده است رفع کنیم.
@Database(entities = arrayOf(DashboardCircularCategory::class,
version = 2)
@TypeConverters(Converters::class)
abstract class ApplicationDatabase : RoomDatabase() {
مورد دوم هیچگونه migration ای ارائه نشده است
اکنون نسخه پایگاهداده را افزایش دادهایم و و سعی میکنیم به پایگاهداده دسترسی داشته باشیم، اپلیکیشن همانند زیر با crash مواجه میشود.
java.lang.IllegalStateException: A migration from 1 to 2 was required but not found.
Please provide the necessary Migration path via RoomDatabase.Builder.addMigration(Migration ...)
or allow for destructive migrations via one of the RoomDatabase.Builder.
fallbackToDestructiveMigration* methods.
Romm یک کلاس به نام Migration دارد، که با تغییرات در شِما توافق کرده است، که مسئول بروزرسانی SQLite است. بنابراین اگر هر تغییری در شِما ایجاد شود و هیچگونه migrationای ارائه نشود، اپلیکیشن crash خواهد کرد.
مورد سوم مهاجرت مخرب
این نوع خطا کاملا واضح است، ما باید یا مهاجرت را ارائه کنیم و یا از fallbackToDestructiveMigration در سازندهی پایگاهداده room استفاده کنیم. برای استفاده سادهتر، بیایید مانند زیر از آن استفاده کنیم:
Room.databaseBuilder(context.getApplicationContext(),
ApplicationDatabase::class.java, "applicationDatabase.db")
.fallbackToDestructiveMigration()
.build()
بنابراین وقتی که میخواهید به پایگاهداده دسترسی داشته باشید، همانطور که نسخه را ارتقا دادهاید، room را در migration بررسی میکند. هیچ migrationای پیدا نمیکند، بنابراین حالت fallbackToDestructiveMigration همهی جدولها را حذف میکند و شناسهی hash شده را اضافه میکند.
نتیجه این خواهد بود که اپلیکیشن crash نمیکند ولی کاربر همهی دادههایش را از دست میدهد.
مهاجرت همراه با تغییر شِما
بیایید موقعیتی را در نظر بگیریم که نیاز داریم دادهها را نگه داریم و همچنین شِمای جدیدی را اضافه کنیم. اینجاست که مفهوم migration آشکار میشود.
بیایید کارهایی که نیاز به انجام است را قدم به قدم انجام دهیم.
1- افزایش نسخهی پایگاهداده به 3
@Database(entities = arrayOf(DashboardCircularCategory::class,
version = 2)
@TypeConverters(Converters::class)
abstract class ApplicationDatabase : RoomDatabase()
2- ایجاد migration از نسخه 2 به 3، که شامل همهی تغییراتی که از نسخه 2 به 3 داشتهاید میباشد.
val MIGRATION_2_3: Migration = object : Migration(2, 3) {
override fun migrate(database: SupportSQLiteDatabase)
database.execSQL("ALTER TABLE `Movies`"
+ " ADD COLUMN rating TEXT")
}
}
3- migration را همانند زیر به سازنده پایگاهدادهی room اضافه کنید.
Room.databaseBuilder(context.getApplicationContext(),
ApplicationDatabase::class.java, "applicationDatabase.db")
.addMigrations(MIGRATION_2_3)
.build()
اکنون، اگر سعی در دسترسی به پایگاهداده room داشته باشد:
- تغییرات بر روی جدول اعمال میشود
- Identity_hash بروزرسانی میشود
- بعد از آن سعی میشود پایگاهداده باز شود. بخاطر آنکه شناسهی hash شده نسخه کنونی و نسخه ذخیره شده یکی هستند. بنابراین مشکلی ایجاد نخواهد شد.
نتیجه
در این مقاله سعی شد مواردی که در migration کردن در پایگاهداده room وجود دارد بررسی شود امیدوارم براتون مفید واقع شده باشد.
دیدگاه و پرسش
در حال دریافت نظرات از سرور، لطفا منتظر بمانید
در حال دریافت نظرات از سرور، لطفا منتظر بمانید