数据库 Room

Room 在 SQLite 上提供了一个抽象层,以便在充分利用 SQLite 的强大功能的同时,能够流畅地访问数据库。

1.Room 入门

1.gradle添加依赖

build.gradle(Module中的gradle)
Groovy

dependencies {
   
    def room_version = "2.2.5"
    implementation "androidx.room:room-runtime:$room_version"
    annotationProcessor "androidx.room:room-compiler:$room_version" // For Kotlin use kapt instead of annotationProcessor  这里 把  annotationProcessor 替换为 kapt
    // optional - Kotlin Extensions and Coroutines support for Room
    implementation "androidx.room:room-ktx:$room_version"
    // optional - RxJava support for Room
    implementation "androidx.room:room-rxjava2:$room_version"
    // optional - Guava support for Room, including Optional and ListenableFuture
    implementation "androidx.room:room-guava:$room_version"
    // Test helpers
    testImplementation "androidx.room:room-testing:$room_version"
}

如果使用kotlin :在 gradle的顶部 添加apply plugin: ‘kotlin-kapt’
build.gradle(Module中的gradle)

apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'
apply plugin: 'kotlin-kapt'
2.Room重要组件介绍

数据库:包含数据库持有者,并作为应用已保留的持久关系型数据的底层连接的主要接入点,使用 @Database 注释的类是:
扩展RoomDatabase的抽象类
在注释中添加与数据库关联的实体列表
包含具有0个参数且返回使用@Dao注释的类的抽象方法 运行时,通过 Room.databasebaseBuilder()或Room.inMemoryDatabaseBuilder()获取Database的实例
Entity:标识数据库中的表

3.Room 数据库入门级使用

创建实体类(数据表 table)

@Entity()
data class User( 
    // 主键 唯一 autoGenerate = true 表示自增
    @PrimaryKey(autoGenerate = true) val uid: Int?,
    // 表头名
    @ColumnInfo(name = "first_name")
    val firstName: String?,
    // 表头名
    @ColumnInfo(name = "last_name")
    val lastName: String?,
    @ColumnInfo(name = "age")
    val age: Int? = 0,
    @ColumnInfo(name = "city")
    val city: String?
)

创建Dao(操作接口)

@Dao
interface UserDao {
    @Query("SELECT * FROM user")
    fun getAll(): List<User>
    @Query("SELECT * FROM user WHERE uid IN (:userIds)")
    fun loadAllByIds(userIds: IntArray): List<User>
    @Query("SELECT * FROM user WHERE age LIKE :age")
    fun findByAge(age: Int): User
    @Insert
    fun insertAll(vararg users: User)
    @Delete
    fun delete(user: User)
}

创建 AppDatabase(名字自取)

@Database(entities = [User::class], version = 1)
abstract class AppDatabase : RoomDatabase() {
    abstract fun userDao(): UserDao
}

初始化

val db = Room.databaseBuilder(
            applicationContext,
            AppDatabase::class.java, "database-name.db"
        ).build()

AndroidStudio Build -> Rebuild Project 成功之后,就会看到 app/build/generated/source/kapt/debug/包名/ 目录下 AppDatabase_Impl和 UserDao_Impl文件。
至此 数据库创建就完成了。
简单调用 Dao的方法

因为不能在 主线程使用,这里使用协程,也可以使用子线程
 GlobalScope.launch {
            db.userDao().insertAll(User(null, "jack", "love", 10, "AA"))
            val all = db.userDao().getAll()
            Log.d("db_db", "all$all")
        }

2.room 常用基础方法举例

对上述进行简单封装
object RoomUtil {
    private val db = Room.databaseBuilder(
        MyAppLication.context,
        AppDatabase::class.java, "database-name.db"
    ).build()
    private val userDao = db.userDao()
    fun getUserDao(): UserDao {
        return userDao
    }
}
@Dao
interface UserDao {
    /**
     * 增
     */
    @Insert()
    fun insert(user: User) // 增加/插入 对象(一行)
    @Insert
    fun insert(users:List<User>) // 增加/插入 对象集合(多行)
    @Insert
    fun insert(vararg user: User) // 增加/插入 n个对象(一行或者多行)
    /**
     * 删
     */
    @Delete
    fun delete(user: User) // 删除 对象(一行)
    @Delete
    fun delete(users: List<User>) // 删除多个对象 (多行)
    @Delete
    fun delete(vararg user: User) // 删除 n个对象(一行或者多行)
    /**
     * 改
     */
    @Update
    fun update(user: User)
    /**
     * 查
     */
    @Query("SELECT * FROM User")
    fun getAll():List<User>
}
记住,每次修改后,都需要rebuild一下
调用:
        GlobalScope.launch(Dispatchers.IO) {
            val all = RoomUtil.getUserDao().getAll()
        }

3.room 数据库 sql语法在room中的使用

在room中的使用(主要以查询为例)如下

@Dao
interface UserDao {
    /**
     * 查
     */
    @Query("SELECT * FROM User")
    fun getAll(): List<User>
    /**
     * 查询 first_name这一列的数据
     */
    @Query("SELECT first_name FROM User")
    fun queryFirstName(): List<String>
    /**
     * 查询 多列 first_name 和 last_name
     */
    @Query("SELECT first_name,last_name From User")
    fun queryManyColumn(): List<User>
    /**
     * 增加查询条件 比如 age的条件 传入参数 在 sql语句中,以 :age 的方式来使用
     */
    @Query("SELECT * FROM user WHERE age=:age")
    fun queryUserByAge(age: Int): List<User>
    @Query("SELECT * FROM User WHERE age BETWEEN :minAge AND :maxAge")
    fun queryUserByAge(minAge: Int, maxAge: Int): List<User>
    @Query("SELECT * FROM User WHERE age IN (:ages)")
    fun queryUserByAge(ages: List<Int>): List<User>
}

在room中其他方法都可以用Query替代,所以,只要sql语句熟悉,可以就使用@Query(“sql语句”)

@Dao
interface UserDao {
    @Query("INSERT INTO User VALUES (:uid,:firstName,:lastName,:age,:city)")
    fun insertByQ(uid: Int?,firstName:String,lastName:String,age: Int,city:String)
    }

4.room 数据库 升级

数据库升级归纳为两类

1.数据库增加表

第一步:创建实体类(数据表)和CarDao

@Entity()
data class Car(
    
    @PrimaryKey(autoGenerate = true)
    var id: Int?,
    @ColumnInfo(name = "car_name")
    var carName: String?,
    @ColumnInfo(name = "value")
    var value: Float? = 0f,
    @ColumnInfo(name = "origin")
    var origin: String?
)
@Dao
interface CarDao {
    @Insert
    fun insert(car: Car)
    @Query("SELECT * FROM Car")
    fun getCars():List<Car>
}

第二步:通过 Migration 类进行增量升级

注意: 下面的sql语句中的字段,名字 类型 一定要和 Entity Car中的属性完全对应,否则会报错 修改时,根据报错,将属性和字段修改为一致即可
    val MIGRATION_1_2 = object : Migration(1, 2) {
        override fun migrate(database: SupportSQLiteDatabase) {
            database.execSQL("CREATE TABLE `Car` (`car_name` TEXT, 'value' REAL, 'origin' TEXT, `id` INTEGER PRIMARY KEY AUTOINCREMENT)")
        }
    }

第三步

private val db = Room.databaseBuilder(
        MyAppLication.context,
        AppDatabase::class.java, "database-name.db")
        .addMigrations(MIGRATION_1_2)
        .build()

第四步:修改AppDatabase

增加 Car class,修改版本号,增加 CarDao关联
@Database(entities = [User::class,Car::class], version = 2)
abstract class AppDatabase : RoomDatabase() {
    abstract fun userDao(): UserDao
    abstract fun carDao(): CarDao
}

RoomUtil完整代码

object RoomUtil {
    val MIGRATION_1_2 = object : Migration(1, 2) {
        override fun migrate(database: SupportSQLiteDatabase) {
            database.execSQL("CREATE TABLE `Car` (`car_name` TEXT, 'value' REAL, 'origin' TEXT, `id` INTEGER PRIMARY KEY AUTOINCREMENT)")
        }
    }
    private val db = Room.databaseBuilder(
        MyAppLication.context,
        AppDatabase::class.java, "database-name.db")
        .addMigrations(MIGRATION_1_2)
        .build()
    private val userDao = db.userDao()
    private val carDao = db.carDao()
    fun getUserDao(): UserDao {
        return userDao
    }
    fun getCarDao(): CarDao {
        return carDao
    }
}
2.数据表增加列

第一步:修改实体类(数据表)和UserDao

@Entity()
data class User(
    // 主键 唯一
    @PrimaryKey(autoGenerate = true) val uid: Int?,
    // 表头名
    @ColumnInfo(name = "first_name")
    var firstName: String?,
    // 表头名
    @ColumnInfo(name = "last_name")
    var lastName: String?,
    @ColumnInfo(name = "age")
    var age: Int? = 0,
    @ColumnInfo(name = "city")
    var city: String?,
    // 新增字段(新增列)
    @ColumnInfo(name = "cn_name")
    var cnName:String?
)
// 受影响的方法,进行相应修改,如果没有,则不需要
  @Query("INSERT INTO User VALUES (:uid,:firstName,:lastName,:age,:city,null)")
    fun insertByQ(uid: Int?,firstName:String,lastName:String,age: Int,city:String)

第二步:修改AppDatabase

增加版本号
@Database(entities = [User::class,Car::class], version = 3)
abstract class AppDatabase : RoomDatabase() {
    abstract fun userDao(): UserDao
    abstract fun carDao(): CarDao
}

第三步:增加Migration

val MIGATION_2_3 = object :Migration(2,3){
        override fun migrate(database: SupportSQLiteDatabase) {
            database.execSQL("ALTER TABLE User ADD COLUMN cn_name TEXT")
        }
    }
    
        private val db = Room.databaseBuilder(
        MyAppLication.context,
        AppDatabase::class.java, "database-name.db")
        .addMigrations(MIGRATION_1_2)
        .addMigrations(MIGATION_2_3)
        .build()

RoomUtil完整代码

object RoomUtil {
    val MIGRATION_1_2 = object : Migration(1, 2) {
        override fun migrate(database: SupportSQLiteDatabase) {
            database.execSQL("CREATE TABLE `Car` (`car_name` TEXT, 'value' REAL, 'origin' TEXT, `id` INTEGER PRIMARY KEY AUTOINCREMENT)")
        }
    }
    val MIGATION_2_3 = object :Migration(2,3){
        override fun migrate(database: SupportSQLiteDatabase) {
            database.execSQL("ALTER TABLE User ADD COLUMN cn_name TEXT")
        }
    }
    private val db = Room.databaseBuilder(
        MyAppLication.context,
        AppDatabase::class.java, "database-name.db")
        .addMigrations(MIGRATION_1_2)
        .addMigrations(MIGATION_2_3)
        .build()
    private val userDao = db.userDao()
    private val carDao = db.carDao()
    fun getUserDao(): UserDao {
        return userDao
    }
    fun getCarDao(): CarDao {
        return carDao
    }
}