迁移 Room 数据库

随着您在应用中添加和更改功能,您需要修改 Room 实体类和底层数据库表以反映这些更改。当应用更新更改数据库模式时,保留已存在于设备数据库中的用户数据非常重要。

Room 支持自动化和手动两种增量迁移选项。自动迁移适用于大多数基本的模式更改,但对于更复杂的更改,您可能需要手动定义迁移路径。

自动迁移

要在两个数据库版本之间声明自动迁移,请将@AutoMigration 注解添加到autoMigrations 属性中,该属性位于@Database 中。

Kotlin

// Database class before the version update.
@Database(
  version = 1,
  entities = [User::class]
)
abstract class AppDatabase : RoomDatabase() {
  ...
}

// Database class after the version update.
@Database(
  version = 2,
  entities = [User::class],
  autoMigrations = [
    AutoMigration (from = 1, to = 2)
  ]
)
abstract class AppDatabase : RoomDatabase() {
  ...
}

Java

// Database class before the version update.
@Database(
  version = 1,
  entities = {User.class}
)
public abstract class AppDatabase extends RoomDatabase {
  ...
}

// Database class after the version update.
@Database(
  version = 2,
  entities = {User.class},
  autoMigrations = {
    @AutoMigration (from = 1, to = 2)
  }
)
public abstract class AppDatabase extends RoomDatabase {
  ...
}

自动迁移规范

如果 Room 检测到不明确的模式更改,并且它无法在没有更多输入的情况下生成迁移计划,则它会抛出编译时错误,并要求您实现AutoMigrationSpec。最常见的情况是,迁移涉及以下一项:

  • 删除或重命名表。
  • 删除或重命名列。

您可以使用AutoMigrationSpec 为 Room 提供它正确生成迁移路径所需的其他信息。在您的RoomDatabase 类中定义一个实现AutoMigrationSpec 的静态类,并使用以下一个或多个对其进行注释:

要将AutoMigrationSpec 实现用于自动迁移,请在相应的@AutoMigration 注解中设置spec 属性。

Kotlin

@Database(
  version = 2,
  entities = [User::class],
  autoMigrations = [
    AutoMigration (
      from = 1,
      to = 2,
      spec = AppDatabase.MyAutoMigration::class
    )
  ]
)
abstract class AppDatabase : RoomDatabase() {
  @RenameTable(fromTableName = "User", toTableName = "AppUser")
  class MyAutoMigration : AutoMigrationSpec
  ...
}

Java

@Database(
  version = 2,
  entities = {AppUser.class},
  autoMigrations = {
    @AutoMigration (
      from = 1,
      to = 2,
      spec = AppDatabase.MyAutoMigration.class
    )
  }
)
public abstract class AppDatabase extends RoomDatabase {
  @RenameTable(fromTableName = "User", toTableName = "AppUser")
  static class MyAutoMigration implements AutoMigrationSpec { }
  ...
}

如果您的应用需要在自动迁移完成后执行更多工作,您可以实现onPostMigrate()。如果您在AutoMigrationSpec 中实现了此方法,则 Room 会在自动迁移完成后调用它。

手动迁移

在迁移涉及复杂模式更改的情况下,Room 可能无法自动生成合适的迁移路径。例如,如果您决定将表中的数据拆分为两个表,Room 无法判断如何执行此拆分。在这种情况下,您必须通过实现Migration 类来手动定义迁移路径。

一个Migration 类通过重写Migration.migrate() 方法明确定义startVersionendVersion 之间的迁移路径。使用addMigrations() 方法将您的Migration 类添加到您的数据库构建器。

Kotlin

val MIGRATION_1_2 = object : Migration(1, 2) {
  override fun migrate(database: SupportSQLiteDatabase) {
    database.execSQL("CREATE TABLE `Fruit` (`id` INTEGER, `name` TEXT, " +
      "PRIMARY KEY(`id`))")
  }
}

val MIGRATION_2_3 = object : Migration(2, 3) {
  override fun migrate(database: SupportSQLiteDatabase) {
    database.execSQL("ALTER TABLE Book ADD COLUMN pub_year INTEGER")
  }
}

Room.databaseBuilder(applicationContext, MyDb::class.java, "database-name")
  .addMigrations(MIGRATION_1_2, MIGRATION_2_3).build()

Java

static final Migration MIGRATION_1_2 = new Migration(1, 2) {
  @Override
  public void migrate(SupportSQLiteDatabase database) {
    database.execSQL("CREATE TABLE `Fruit` (`id` INTEGER, "
      + "`name` TEXT, PRIMARY KEY(`id`))");
  }
};

static final Migration MIGRATION_2_3 = new Migration(2, 3) {
  @Override
  public void migrate(SupportSQLiteDatabase database) {
    database.execSQL("ALTER TABLE Book "
      + " ADD COLUMN pub_year INTEGER");
  }
};

Room.databaseBuilder(getApplicationContext(), MyDb.class, "database-name")
  .addMigrations(MIGRATION_1_2, MIGRATION_2_3).build();

定义迁移路径时,您可以对某些版本使用自动迁移,对其他版本使用手动迁移。如果您为同一版本同时定义自动迁移和手动迁移,则 Room 将使用手动迁移。

测试迁移

迁移通常很复杂,错误定义的迁移可能会导致您的应用崩溃。为了保持应用的稳定性,请测试您的迁移。Room 提供了一个room-testing Maven 工件来帮助自动化和手动迁移的测试过程。要使此工件正常工作,您必须首先导出数据库的模式。

导出模式

Room可以在编译时将数据库的模式信息导出到JSON文件中。导出的JSON文件代表数据库的模式历史记录。将这些文件存储在版本控制系统中,以便Room可以创建数据库的较低版本以进行测试,并启用自动迁移生成。

使用Room Gradle插件设置模式位置

如果您使用的是Room 2.6.0或更高版本,您可以应用Room Gradle插件并使用room扩展名来指定模式目录。

Groovy

plugins {
  id 'androidx.room'
}

room {
  schemaDirectory "$projectDir/schemas"
}

Kotlin

plugins {
  id("androidx.room")
}

room {
  schemaDirectory("$projectDir/schemas")
}

如果您的数据库模式根据变体、风格或构建类型而有所不同,则必须使用schemaDirectory()配置多次指定不同的位置,每次都使用variantMatchName作为第一个参数。每个配置可以根据与变体名称的简单比较匹配一个或多个变体。

确保这些配置是详尽的,并涵盖所有变体。您还可以包含一个不带variantMatchNameschemaDirectory()来处理任何其他配置都不匹配的变体。例如,在一个具有两个构建风格demofull以及两个构建类型debugrelease的应用程序中,以下配置是有效的

Groovy

room {
  // Applies to 'demoDebug' only
  schemaDirectory "demoDebug", "$projectDir/schemas/demoDebug"

  // Applies to 'demoDebug' and 'demoRelease'
  schemaDirectory "demo", "$projectDir/schemas/demo"

  // Applies to 'demoDebug' and 'fullDebug'
  schemaDirectory "debug", "$projectDir/schemas/debug"

  // Applies to variants that aren't matched by other configurations.
  schemaDirectory "$projectDir/schemas"
}

Kotlin

room {
  // Applies to 'demoDebug' only
  schemaDirectory("demoDebug", "$projectDir/schemas/demoDebug")

  // Applies to 'demoDebug' and 'demoRelease'
  schemaDirectory("demo", "$projectDir/schemas/demo")

  // Applies to 'demoDebug' and 'fullDebug'
  schemaDirectory("debug", "$projectDir/schemas/debug")

  // Applies to variants that aren't matched by other configurations.
  schemaDirectory("$projectDir/schemas")
}

使用注解处理器选项设置模式位置

如果您使用的是Room 2.5.2或更低版本,或者您没有使用Room Gradle插件,请使用room.schemaLocation注解处理器选项设置模式位置。

此目录中的文件用作某些Gradle任务的输入和输出。为了确保增量构建和缓存构建的正确性和性能,您必须使用Gradle的CommandLineArgumentProvider来告知Gradle有关此目录的信息。

首先,将下面显示的RoomSchemaArgProvider类复制到模块的Gradle构建文件中。asArguments()方法在示例类中将room.schemaLocation=${schemaDir.path}传递给KSP。如果您使用的是KAPTjavac,请将此值更改为-Aroom.schemaLocation=${schemaDir.path}

Groovy

class RoomSchemaArgProvider implements CommandLineArgumentProvider {

  @InputDirectory
  @PathSensitive(PathSensitivity.RELATIVE)
  File schemaDir

  RoomSchemaArgProvider(File schemaDir) {
    this.schemaDir = schemaDir
  }

  @Override
  Iterable<String> asArguments() {
    // Note: If you're using KAPT and javac, change the line below to
    // return ["-Aroom.schemaLocation=${schemaDir.path}".toString()].
    return ["room.schemaLocation=${schemaDir.path}".toString()]
  }
}

Kotlin

class RoomSchemaArgProvider(
  @get:InputDirectory
  @get:PathSensitive(PathSensitivity.RELATIVE)
  val schemaDir: File
) : CommandLineArgumentProvider {

  override fun asArguments(): Iterable<String> {
    // Note: If you're using KAPT and javac, change the line below to
    // return listOf("-Aroom.schemaLocation=${schemaDir.path}").
    return listOf("room.schemaLocation=${schemaDir.path}")
  }
}

然后配置编译选项以使用具有指定模式目录的RoomSchemaArgProvider

Groovy

// For KSP, configure using KSP extension:
ksp {
  arg(new RoomSchemaArgProvider(new File(projectDir, "schemas")))
}

// For javac or KAPT, configure using android DSL:
android {
  ...
  defaultConfig {
    javaCompileOptions {
      annotationProcessorOptions {
        compilerArgumentProviders(
          new RoomSchemaArgProvider(new File(projectDir, "schemas"))
        )
      }
    }
  }
}

Kotlin

// For KSP, configure using KSP extension:
ksp {
  arg(RoomSchemaArgProvider(File(projectDir, "schemas")))
}

// For javac or KAPT, configure using android DSL:
android {
  ...
  defaultConfig {
    javaCompileOptions {
      annotationProcessorOptions {
        compilerArgumentProviders(
          RoomSchemaArgProvider(File(projectDir, "schemas"))
        )
      }
    }
  }
}

测试单个迁移

在测试迁移之前,请将Room中的androidx.room:room-testing Maven构件添加到您的测试依赖项中,并将导出的模式的位置添加为资产文件夹

build.gradle

Groovy

android {
    ...
    sourceSets {
        // Adds exported schema location as test app assets.
        androidTest.assets.srcDirs += files("$projectDir/schemas".toString())
    }
}

dependencies {
    ...
    androidTestImplementation "androidx.room:room-testing:2.6.1"
}

Kotlin

android {
    ...
    sourceSets {
        // Adds exported schema location as test app assets.
        getByName("androidTest").assets.srcDir("$projectDir/schemas")
    }
}

dependencies {
    ...
    testImplementation("androidx.room:room-testing:2.6.1")
}

测试包提供了一个MigrationTestHelper类,该类可以读取导出的模式文件。该包还实现了JUnit4 TestRule接口,因此它可以管理创建的数据库。

以下示例演示了单个迁移的测试

Kotlin

@RunWith(AndroidJUnit4::class)
class MigrationTest {
    private val TEST_DB = "migration-test"

    @get:Rule
    val helper: MigrationTestHelper = MigrationTestHelper(
            InstrumentationRegistry.getInstrumentation(),
            MigrationDb::class.java.canonicalName,
            FrameworkSQLiteOpenHelperFactory()
    )

    @Test
    @Throws(IOException::class)
    fun migrate1To2() {
        var db = helper.createDatabase(TEST_DB, 1).apply {
            // Database has schema version 1. Insert some data using SQL queries.
            // You can't use DAO classes because they expect the latest schema.
            execSQL(...)

            // Prepare for the next version.
            close()
        }

        // Re-open the database with version 2 and provide
        // MIGRATION_1_2 as the migration process.
        db = helper.runMigrationsAndValidate(TEST_DB, 2, true, MIGRATION_1_2)

        // MigrationTestHelper automatically verifies the schema changes,
        // but you need to validate that the data was migrated properly.
    }
}

Java

@RunWith(AndroidJUnit4.class)
public class MigrationTest {
    private static final String TEST_DB = "migration-test";

    @Rule
    public MigrationTestHelper helper;

    public MigrationTest() {
        helper = new MigrationTestHelper(InstrumentationRegistry.getInstrumentation(),
                MigrationDb.class.getCanonicalName(),
                new FrameworkSQLiteOpenHelperFactory());
    }

    @Test
    public void migrate1To2() throws IOException {
        SupportSQLiteDatabase db = helper.createDatabase(TEST_DB, 1);

        // Database has schema version 1. Insert some data using SQL queries.
        // You can't use DAO classes because they expect the latest schema.
        db.execSQL(...);

        // Prepare for the next version.
        db.close();

        // Re-open the database with version 2 and provide
        // MIGRATION_1_2 as the migration process.
        db = helper.runMigrationsAndValidate(TEST_DB, 2, true, MIGRATION_1_2);

        // MigrationTestHelper automatically verifies the schema changes,
        // but you need to validate that the data was migrated properly.
    }
}

测试所有迁移

虽然可以测试单个增量迁移,但我们建议您包含一个涵盖应用程序数据库定义的所有迁移的测试。这有助于确保最近创建的数据库实例与遵循已定义迁移路径的旧实例之间没有差异。

以下示例演示了所有已定义迁移的测试

Kotlin

@RunWith(AndroidJUnit4::class)
class MigrationTest {
    private val TEST_DB = "migration-test"

    // Array of all migrations.
    private val ALL_MIGRATIONS = arrayOf(
            MIGRATION_1_2, MIGRATION_2_3, MIGRATION_3_4)

    @get:Rule
    val helper: MigrationTestHelper = MigrationTestHelper(
            InstrumentationRegistry.getInstrumentation(),
            AppDatabase::class.java.canonicalName,
            FrameworkSQLiteOpenHelperFactory()
    )

    @Test
    @Throws(IOException::class)
    fun migrateAll() {
        // Create earliest version of the database.
        helper.createDatabase(TEST_DB, 1).apply {
            close()
        }

        // Open latest version of the database. Room validates the schema
        // once all migrations execute.
        Room.databaseBuilder(
            InstrumentationRegistry.getInstrumentation().targetContext,
            AppDatabase::class.java,
            TEST_DB
        ).addMigrations(*ALL_MIGRATIONS).build().apply {
            openHelper.writableDatabase.close()
        }
    }
}

Java

@RunWith(AndroidJUnit4.class)
public class MigrationTest {
    private static final String TEST_DB = "migration-test";

    @Rule
    public MigrationTestHelper helper;

    public MigrationTest() {
        helper = new MigrationTestHelper(InstrumentationRegistry.getInstrumentation(),
                AppDatabase.class.getCanonicalName(),
                new FrameworkSQLiteOpenHelperFactory());
    }

    @Test
    public void migrateAll() throws IOException {
        // Create earliest version of the database.
        SupportSQLiteDatabase db = helper.createDatabase(TEST_DB, 1);
        db.close();

        // Open latest version of the database. Room validates the schema
        // once all migrations execute.
        AppDatabase appDb = Room.databaseBuilder(
                InstrumentationRegistry.getInstrumentation().getTargetContext(),
                AppDatabase.class,
                TEST_DB)
                .addMigrations(ALL_MIGRATIONS).build();
        appDb.getOpenHelper().getWritableDatabase();
        appDb.close();
    }

    // Array of all migrations.
    private static final Migration[] ALL_MIGRATIONS = new Migration[]{
            MIGRATION_1_2, MIGRATION_2_3, MIGRATION_3_4};
}

优雅地处理缺少的迁移路径

如果Room找不到迁移路径将设备上的现有数据库升级到当前版本,则会发生IllegalStateException。如果在缺少迁移路径时丢失现有数据是可以接受的,则在创建数据库时调用fallbackToDestructiveMigration()构建器方法

Kotlin

Room.databaseBuilder(applicationContext, MyDb::class.java, "database-name")
        .fallbackToDestructiveMigration()
        .build()

Java

Room.databaseBuilder(getApplicationContext(), MyDb.class, "database-name")
        .fallbackToDestructiveMigration()
        .build();

此方法告诉Room在需要执行增量迁移并且没有定义迁移路径时,破坏性地重新创建应用程序数据库中的表。

如果您只想在某些情况下让Room回退到破坏性重建,则有一些替代fallbackToDestructiveMigration()的方法

处理升级到Room 2.2.0时的列默认值

在Room 2.2.0和更高版本中,您可以使用注解@ColumnInfo(defaultValue = "...")定义列的默认值。在低于2.2.0的版本中,定义列默认值的唯一方法是直接在执行的SQL语句中定义它,这会创建一个Room不知道的默认值。这意味着,如果数据库最初是由低于2.2.0版本的Room创建的,则将应用程序升级到使用Room 2.2.0可能需要您为使用Room API之外定义的现有默认值提供特殊的迁移路径。

例如,假设数据库的版本1定义了一个Song实体

Kotlin

// Song entity, database version 1, Room 2.1.0.
@Entity
data class Song(
    @PrimaryKey
    val id: Long,
    val title: String
)

Java

// Song entity, database version 1, Room 2.1.0.
@Entity
public class Song {
    @PrimaryKey
    final long id;
    final String title;
}

还假设相同数据库的版本2添加了一个新的NOT NULL列,并定义了从版本1到版本2的迁移路径

Kotlin

// Song entity, database version 2, Room 2.1.0.
@Entity
data class Song(
    @PrimaryKey
    val id: Long,
    val title: String,
    val tag: String // Added in version 2.
)

// Migration from 1 to 2, Room 2.1.0.
val MIGRATION_1_2 = object : Migration(1, 2) {
    override fun migrate(database: SupportSQLiteDatabase) {
        database.execSQL(
            "ALTER TABLE Song ADD COLUMN tag TEXT NOT NULL DEFAULT ''")
    }
}

Java

// Song entity, database version 2, Room 2.1.0.
@Entity
public class Song {
    @PrimaryKey
    final long id;
    final String title;
    @NonNull
    final String tag; // Added in version 2.
}


// Migration from 1 to 2, Room 2.1.0.
static final Migration MIGRATION_1_2 = new Migration(1, 2) {
    @Override
    public void migrate(SupportSQLiteDatabase database) {
        database.execSQL(
            "ALTER TABLE Song ADD COLUMN tag TEXT NOT NULL DEFAULT ''");
    }
};

这会导致更新和应用程序的新安装之间的基础表出现差异。因为tag列的默认值仅在从版本1到版本2的迁移路径中声明,所以从版本2开始安装应用程序的任何用户在其数据库模式中都没有tag的默认值。

在低于2.2.0版本的Room中,这种差异是无害的。但是,如果应用程序稍后升级到使用Room 2.2.0或更高版本,并更改Song实体类以使用@ColumnInfo注解包含tag的默认值,则Room可以检测到这种差异。这会导致模式验证失败。

为了帮助确保在较早的迁移路径中声明列默认值时所有用户的数据库模式保持一致,请在首次升级应用程序以使用Room 2.2.0或更高版本时执行以下操作

  1. 使用@ColumnInfo注解在其各自的实体类中声明列默认值。
  2. 将数据库版本号增加1。
  3. 定义一个迁移路径到新版本,该路径实现删除和重新创建策略,以便将必要的默认值添加到现有列。

以下示例演示了此过程

Kotlin

// Migration from 2 to 3, Room 2.2.0.
val MIGRATION_2_3 = object : Migration(2, 3) {
    override fun migrate(database: SupportSQLiteDatabase) {
        database.execSQL("""
                CREATE TABLE new_Song (
                    id INTEGER PRIMARY KEY NOT NULL,
                    name TEXT,
                    tag TEXT NOT NULL DEFAULT ''
                )
                """.trimIndent())
        database.execSQL("""
                INSERT INTO new_Song (id, name, tag)
                SELECT id, name, tag FROM Song
                """.trimIndent())
        database.execSQL("DROP TABLE Song")
        database.execSQL("ALTER TABLE new_Song RENAME TO Song")
    }
}

Java

// Migration from 2 to 3, Room 2.2.0.
static final Migration MIGRATION_2_3 = new Migration(2, 3) {
    @Override
    public void migrate(SupportSQLiteDatabase database) {
        database.execSQL("CREATE TABLE new_Song (" +
                "id INTEGER PRIMARY KEY NOT NULL," +
                "name TEXT," +
                "tag TEXT NOT NULL DEFAULT '')");
        database.execSQL("INSERT INTO new_Song (id, name, tag) " +
                "SELECT id, name, tag FROM Song");
        database.execSQL("DROP TABLE Song");
        database.execSQL("ALTER TABLE new_Song RENAME TO Song");
    }
};