在 Android 项目中使用 Room 可以通过以下步骤实现:
1、添加 Room 的依赖
在项目的 build.gradle 文件中添加以下依赖:
dependencies {
...
implementation "androidx.room:room-runtime:<version>"
kapt "androidx.room:room-compiler:<version>"
}
2、创建 Entity
使用 @Entity 注解创建一个数据表实体类,其中可以使用 @PrimaryKey 注解设置主键,@ColumnInfo 注解设置列名等。
例如:
@Entity(tableName = "users")
data class User(
@PrimaryKey val id: Int,
@ColumnInfo(name = "name") val name: String,
@ColumnInfo(name = "email") val email: String
)
3、创建 DAO
使用 @Dao 注解创建数据访问对象,定义访问数据库的方法。可以使用 @Query 注解执行 SQL 查询,也可以使用其他注解进行数据操作。
例如:
@Dao
interface UserDao {
@Query("SELECT * FROM users")
fun getAll(): List<User>
@Insert
fun insertAll(vararg users: User)
}
4、创建 Room 数据库
使用 @Database 注解创建 Room 数据库,并指定包含 Entity 的数组和版本号。
例如:
@Database(entities = [User::class], version = 1)
abstract class AppDatabase : RoomDatabase() {
abstract fun userDao(): UserDao
}
5、初始化 Room 数据库
在 Application 类中创建 Room 数据库的实例。
例如:
@Database(entities = [User::class], version = 1)
abstract class AppDatabase : RoomDatabase() {
abstract fun userDao(): UserDao
}
@HiltAndroidApp
class MyApplication : Application() {
val db = Room.databaseBuilder(
applicationContext,
AppDatabase::class.java, "my-database"
).build()
}
6、使用 Room 数据库
通过 Room 数据库的实例访问 DAO,执行数据操作。
例如:
val userDao = (application as MyApplication).db.userDao()
val users = userDao.getAll()
userDao.insertAll(User(1, "Alice", "alice@example.com"))
以上是在 Android 项目中使用 Room 的基本步骤,可以根据实际需求进行调整。
除了基本的使用方法之外,Room 还提供了以下功能:
(1)、使用 LiveData 自动更新 UI
可以在 DAO 的查询方法上使用 LiveData 返回数据,从而在数据变化时自动更新 UI。
例如:
@Dao
interface UserDao {
@Query("SELECT * FROM users")
fun getAll(): LiveData<List<User>>
}
在 UI 组件中观察 LiveData,当数据变化时会自动更新 UI。
userViewModel.allUsers.observe(this, Observer { users ->
// Update UI
})
(2)、使用关系映射
可以在 Entity 类中使用 @Relation 注解定义与其他表的关系,从而方便地访问关联数据。
例如:
data class UserWithPets(
@Embedded val user: User,
@Relation(
parentColumn = "id",
entityColumn = "user_id"
)
val pets: List<Pet>
)
在 DAO 中查询数据时,使用 @Transaction 注解可以在单个事务中执行多个操作。
例如:
@Transaction
@Query("SELECT * FROM users WHERE id = :userId")
fun getUserWithPets(userId: Int): LiveData<UserWithPets>
(3)、数据库迁移
当需要更改数据库结构时,可以使用 Room 提供的数据库迁移机制。在 AppDatabase 中使用 @Database 注解时指定 migration 数组,即可在升级版本时执行相应的迁移操作。
例如:
@Database(entities = [User::class], version = 2, exportSchema = false)
abstract class AppDatabase : RoomDatabase() {
...
companion object {
val MIGRATION_1_2 = object : Migration(1, 2) {
override fun migrate(database: SupportSQLiteDatabase) {
database.execSQL("ALTER TABLE users ADD COLUMN age INTEGER NOT NULL DEFAULT 0")
}
}
}
...
}
在创建数据库实例时指定 migration 数组即可进行数据迁移。
例如:
val db = Room.databaseBuilder(
applicationContext,
AppDatabase::class.java,
"my-database"
)
.addMigrations(AppDatabase.MIGRATION_1_2)
.build()
(4)、使用 TypeConverters 进行数据类型转换
在 Entity 类中可以使用 @TypeConverters 注解指定一个或多个类型转换器,将不支持的数据类型转换为 Room 支持的数据类型,或将 Room 支持的数据类型转换为应用程序中的自定义类型。
例如:
@Entity
data class User(
@PrimaryKey val id: Int,
@ColumnInfo(name = "name") val name: String,
@ColumnInfo(name = "birthdate") val birthdate: Date
)
class Converters {
@TypeConverter
fun fromTimestamp(value: Long?): Date? {
return value?.let { Date(it) }
}
@TypeConverter
fun dateToTimestamp(date: Date?): Long? {
return date?.time
}
}
@Database(entities = [User::class], version = 1)
@TypeConverters(Converters::class)
abstract class AppDatabase : RoomDatabase() {
...
}
(5)、使用 RxJava 进行异步操作
在 DAO 中使用 RxJava 的 Observable 或 Single 对象,可以进行异步数据操作,并使用 RxJava 提供的操作符进行数据转换、筛选、组合等操作。
例如:
@Dao
interface UserDao {
@Query("SELECT * FROM users")
fun getAll(): Observable<List<User>>
@Insert
fun insert(user: User): Single<Long>
}
userDao.getAll()
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe { users ->
// Update UI
}
userDao.insert(User(1, "Alice", Date()))
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe { id ->
// Show success message
}
(6)、使用协程进行异步操作
在 DAO 中使用协程的 suspend 函数,可以进行异步数据操作,并使用协程提供的操作符进行数据转换、筛选、组合等操作。
例如:
@Dao
interface UserDao {
@Query("SELECT * FROM users")
suspend fun getAll(): List<User>
@Insert
suspend fun insert(user: User): Long
}
GlobalScope.launch(Dispatchers.IO) {
val users = userDao.getAll()
withContext(Dispatchers.Main) {
// Update UI
}
}
GlobalScope.launch(Dispatchers.IO) {
val id = userDao.insert(User(1, "Alice", Date()))
withContext(Dispatchers.Main) {
// Show success message
}
}