使用 Room 實體定義資料

使用 Room 持續性程式庫儲存應用程式的資料時,您可以定義實體來代表要儲存的物件。每個實體都會對應到相關 Room 資料庫中的一個資料表,而實體的每個例項則代表對應資料表中的一列資料。

這表示您不必編寫任何 SQL 程式碼,就可以使用 Room 實體定義資料庫結構定義

實體剖析

您可以將每個 Room 實體定義為已加上 @Entity 註解的類別。Room 實體包含資料庫中對應資料表內每個資料欄的欄位,包括構成主鍵的一或多個資料欄。

以下程式碼是簡單的實體範例,定義了內含 ID、名字和姓氏資料欄的 User 資料表:

Kotlin

@Entity
data class User(
    @PrimaryKey val id: Int,

    val firstName: String?,
    val lastName: String?
)

Java

@Entity
public class User {
    @PrimaryKey
    public int id;

    public String firstName;
    public String lastName;
}

根據預設,Room 會使用類別名稱做為資料庫的資料表名稱。如果想讓資料表使用不同名稱,請設定 @Entity 註解的 tableName 屬性。同樣地,Room 預設會使用欄位名稱做為資料庫中的資料欄名稱。如果您想讓某個資料欄使用不同名稱,請為相關欄位加上 @ColumnInfo 註解,並設定 name 屬性。以下示範資料表及其資料欄的自訂名稱:

Kotlin

@Entity(tableName = "users")
data class User (
    @PrimaryKey val id: Int,
    @ColumnInfo(name = "first_name") val firstName: String?,
    @ColumnInfo(name = "last_name") val lastName: String?
)

Java

@Entity(tableName = "users")
public class User {
    @PrimaryKey
    public int id;

    @ColumnInfo(name = "first_name")
    public String firstName;

    @ColumnInfo(name = "last_name")
    public String lastName;
}

定義主鍵

每個 Room 實體都必須定義主鍵 用於識別資料庫資料表中每個資料列的專屬 ID。最簡單的做法是使用 @PrimaryKey 為單一資料欄加上註解:

Kotlin

@PrimaryKey val id: Int

Java

@PrimaryKey
public int id;

定義複合式主鍵

如果需要利用多個資料欄的組合來識別實體的執行個體,建議您定義「複合式主鍵」,方法是在 @EntityprimaryKeys 屬性中列出這些資料欄:

Kotlin

@Entity(primaryKeys = ["firstName", "lastName"])
data class User(
    val firstName: String?,
    val lastName: String?
)

Java

@Entity(primaryKeys = {"firstName", "lastName"})
public class User {
    public String firstName;
    public String lastName;
}

忽略欄位

根據預設,Room 會為實體中定義的每個欄位建立一個資料欄。如果實體中有不想保留的欄位,您可以使用 @Ignore 加上註解,如以下程式碼片段所示:

Kotlin

@Entity
data class User(
    @PrimaryKey val id: Int,
    val firstName: String?,
    val lastName: String?,
    @Ignore val picture: Bitmap?
)

Java

@Entity
public class User {
    @PrimaryKey
    public int id;

    public String firstName;
    public String lastName;

    @Ignore
    Bitmap picture;
}

如果實體沿用父系實體的欄位,則使用 @Entity 屬性的 ignoredColumns 屬性通常比較簡單:

Kotlin

open class User {
    var picture: Bitmap? = null
}

@Entity(ignoredColumns = ["picture"])
data class RemoteUser(
    @PrimaryKey val id: Int,
    val hasVpn: Boolean
) : User()

Java

@Entity(ignoredColumns = "picture")
public class RemoteUser extends User {
    @PrimaryKey
    public int id;

    public boolean hasVpn;
}

Room 支援多種註解類型,讓您能夠輕鬆搜尋資料庫中資料表內的詳細資料。除非應用程式的 minSdkVersion 小於 16,否則請使用全文搜尋功能。

支援全文搜尋功能

如果應用程式需要透過全文搜尋 (FTS) 功能快速存取資料庫資訊,請利用虛擬資料表來支援您的實體,該資料表須採用 FTS3 或 FTS4 SQLite 擴充功能模組。如要使用這項適用於 Room 2.1.0 以上版本的功能,請在特定實體中新增 @Fts3@Fts4 註解,如以下程式碼片段所示:

Kotlin

// Use `@Fts3` only if your app has strict disk space requirements or if you
// require compatibility with an older SQLite version.
@Fts4
@Entity(tableName = "users")
data class User(
    /* Specifying a primary key for an FTS-table-backed entity is optional, but
       if you include one, it must use this type and column name. */
    @PrimaryKey @ColumnInfo(name = "rowid") val id: Int,
    @ColumnInfo(name = "first_name") val firstName: String?
)

Java

// Use `@Fts3` only if your app has strict disk space requirements or if you
// require compatibility with an older SQLite version.
@Fts4
@Entity(tableName = "users")
public class User {
    // Specifying a primary key for an FTS-table-backed entity is optional, but
    // if you include one, it must use this type and column name.
    @PrimaryKey
    @ColumnInfo(name = "rowid")
    public int id;

    @ColumnInfo(name = "first_name")
    public String firstName;
}

假如資料表支援多種語言的內容,請使用 languageId 選項來指定儲存各資料列語言資訊的資料欄:

Kotlin

@Fts4(languageId = "lid")
@Entity(tableName = "users")
data class User(
    // ...
    @ColumnInfo(name = "lid") val languageId: Int
)

Java

@Fts4(languageId = "lid")
@Entity(tableName = "users")
public class User {
    // ...

    @ColumnInfo(name = "lid")
    int languageId;
}

Room 提供多種其他選項來定義受 FTS 支援的實體,包括結果排序、權杖化器類型以及當做外部內容管理的資料表。如要進一步瞭解這些選項,請參閱 FtsOptions 參考資料。

索引專用的資料欄

如果應用程式必須支援的 SDK 版本,不允許由 FTS3 或 FTS4 資料表支援的實體,您依然可以為資料庫中的特定資料欄建立索引,以加快查詢速度。如要為實體加入索引,請在 @Entity 註解中加入 indices 屬性,列出要在索引或複合式索引中加入的資料欄名稱。下列程式碼片段示範這項註解程序:

Kotlin

@Entity(indices = [Index(value = ["last_name", "address"])])
data class User(
    @PrimaryKey val id: Int,
    val firstName: String?,
    val address: String?,
    @ColumnInfo(name = "last_name") val lastName: String?,
    @Ignore val picture: Bitmap?
)

Java

@Entity(indices = {@Index("name"),
        @Index(value = {"last_name", "address"})})
public class User {
    @PrimaryKey
    public int id;

    public String firstName;
    public String address;

    @ColumnInfo(name = "last_name")
    public String lastName;

    @Ignore
    Bitmap picture;
}

在某些情況下,資料庫中的特定欄位或欄位群組不得重複。只要將 @Index 註解的 unique 屬性設為 true,就能強制執行這個不重複屬性。下列程式碼範例可防止資料表出現兩個資料列,當中含有相同的 firstNamelastName 資料欄值組合:

Kotlin

@Entity(indices = [Index(value = ["first_name", "last_name"],
        unique = true)])
data class User(
    @PrimaryKey val id: Int,
    @ColumnInfo(name = "first_name") val firstName: String?,
    @ColumnInfo(name = "last_name") val lastName: String?,
    @Ignore var picture: Bitmap?
)

Java

@Entity(indices = {@Index(value = {"first_name", "last_name"},
        unique = true)})
public class User {
    @PrimaryKey
    public int id;

    @ColumnInfo(name = "first_name")
    public String firstName;

    @ColumnInfo(name = "last_name")
    public String lastName;

    @Ignore
    Bitmap picture;
}

加入以 AutoValue 為基礎的物件

在 Room 2.1.0 以上版本中,您可以使用 Java 的不可變值類別 (也就是加上 @AutoValue 註解的類別),做為應用程式資料庫中的實體。當實體的兩個執行個體當中的資料欄含有相同的值時,這項支援尤其實用。

使用加上 @AutoValue 註解的類別做為實體時,您可以使用 @PrimaryKey@ColumnInfo@Embedded@Relation 為類別的抽象方法加上註解。不過,使用這些註解時,您必須每次都加入 @CopyAnnotations 註解,讓 Room 能夠正確解讀方法自動產生的實作項目。

下列程式碼片段是加上 @AutoValue 註解的類別範例,Room 會將此類別視為實體:

User.java

@AutoValue
@Entity
public abstract class User {
    // Supported annotations must include `@CopyAnnotations`.
    @CopyAnnotations
    @PrimaryKey
    public abstract long getId();

    public abstract String getFirstName();
    public abstract String getLastName();

    // Room uses this factory method to create User objects.
    public static User create(long id, String firstName, String lastName) {
        return new AutoValue_User(id, firstName, lastName);
    }
}