使用 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;
定義複合式主鍵
如果需要利用多個資料欄的組合來識別實體的執行個體,建議您定義「複合式主鍵」,方法是在 @Entity
的 primaryKeys
屬性中列出這些資料欄:
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
,就能強制執行這個不重複屬性。下列程式碼範例可防止資料表出現兩個資料列,當中含有相同的 firstName
和 lastName
資料欄值組合:
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 會將此類別視為實體:
@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); } }