闲来无事,总结一下前一阵子对Room数据开源库的学习。
数据库存储是安卓开发中数据存储的方式之一,并且安卓为我们提供了sqlite数据库和SQLiteOpenHelper等等强大的开发支持。而Room,则是对它们的一层封装,为的是让我们更加方便地操作sqlite数据库。
以下,为使用步骤:
加入依赖
implementation "android.arch.persistence.room:runtime:1.0.0"
annotationProcessor "android.arch.persistence.room:compiler:1.0.0"
创建实体类
创建队员类(User)
@Entity(tableName = "users", //表名
primaryKeys = {"id", "name"},//主键
indices = {//索引
@Index(value = "id", unique = true) //唯一性
}) //实体
public class User implements Comparable {
@android.support.annotation.NonNull
@ColumnInfo(name = "id")
private long id;
@android.support.annotation.NonNull
@ColumnInfo(name = "name")
private String name;
@ColumnInfo(name = "position")
private String position;
@Embedded
private UserPerforms performs;
@Ignore
public User() {
}
public User(long id, String name, String position) {
this.id = id;
this.name = name;
this.position = position;
}
// ...get/set+toString()
@Override
public int compareTo(@NonNull Object o) {
if (o instanceof User) {
User u = (User) o;
if (this.id > u.id) {
return 1;
} else if (this.id < u.id) {
return -1;
} else if (this.id == u.id) {
return 0;
}
}
return 0;
}
}
这个类里面的东西我解释一下:
@Entity,表示这个类是数据库中的一张表(实体)
@Embedded,表示在这个表中嵌入另一个类(其实就是嵌了一张表)
@NonNull,主键必须声明非空,而且要在@ColumnInfo上面
@ColumnInfo(name = "id") 表示这个属性是数据库表的某一列,列名是id
@Ignore,让room忽略(我是标在无参构造方法上的,表示不让room用这个构造方法构造对象)
compareTo()方法,这个我是为了保证有序,覆写了一下
创建队员表现类
@Entity(tableName = "performs",
primaryKeys = "p_id", // 主键
foreignKeys = @ForeignKey(entity = User.class
, parentColumns = "id"
, childColumns = "p_id"
, onUpdate = ForeignKey.CASCADE
, onDelete = ForeignKey.CASCADE))
public class UserPerforms {
@android.support.annotation.NonNull
@ColumnInfo(name = "p_id")
private long p_id;
@ColumnInfo(name = "score")
private int score;
@ColumnInfo(name = "assist")
private int assist;
@Ignore
public UserPerforms() {
}
public UserPerforms(long p_id, int score, int assist) {
this.p_id = p_id;
this.score = score;
this.assist = assist;
}
//...get/set+toString()
}
注意里面外键的定义:
foreignKeys = @ForeignKey(entity = User.class
, parentColumns = "id"
, childColumns = "p_id"
, onUpdate = ForeignKey.CASCADE
, onDelete = ForeignKey.CASCADE)
entity:关联表名(类)
parentColumns:父列名
childColumns:子列名
onUpdate,onDelete:父列列值发生变化时,子列的处理方式(我都用的级联,省事儿)
创建数据库类
此时要继承RoomDatabase这个类。这里,我用了下单例
@Database(entities = {User.class, UserPerforms.class}, version = 1, exportSchema = true)
public abstract class DatabaseCreator extends RoomDatabase {
private static DatabaseCreator databaseCreator;
public static DatabaseCreator getInstance(Context context) {
if (databaseCreator == null) {
synchronized (DatabaseCreator.class) {
if (databaseCreator == null) {
databaseCreator = Room.databaseBuilder(context.getApplicationContext(), DatabaseCreator.class,
"user_perfom_2018.db").addMigrations(new Migration(1, 2) {
//版本升级.从1升到2
@Override
public void migrate(@NonNull SupportSQLiteDatabase database) {
database.execSQL("create table coach(id int,name varchar(30),age int,constraint pk_coach PRIMARY_KEY(id) )");
}
//版本升级并不方便,也不提倡频繁对数据库进行组织结构的变动
}).build();
}
}
}
return databaseCreator;
}
public static void onDestroy() {
databaseCreator = null;
}
public abstract CRUDDAO getDao();
}
类的说明:
@Database,说明这是一个room的数据库类。里面的参数:entities说明数据库里有哪些表(实体),version是版本号,exportSchema是是否允许导出,默认为true(导出后的数据库什么样的,我们最后看看)
单例里面Room.databaseBuilder就是构造一个数据库构造者,参数列表是上下文,数据库类名和数据库文件名
addMigrations()是升级数据库时用的,我们在构造完数据库后,每对数据库表的一次结构变化操作,都要调用这个方法升级数据库,并在migrate()方法中执行升级的语句。所以,升级不方便,也别没事儿就改变数据库表的结构。
抽象方法getDao(),这个是获取数据库操作接口的方法,不用我们实现,编译通过后,room会帮我们实现
DAO接口
方才说了获取数据库操作接口,这个接口要我们自己声明,但不用实现。
这个接口,无非就是CRUD增查改删,不同的操作要用上不同的注解。我一个一个说:
声明
声明接口,配上注解@Dao:
@Dao
public interface CRUDDAO {}
查询所有
查询所有User,配上@Query注解,并给注解传入sql语句
@Query("select * from users")
List<User> getAllUsers();
//查询方法返回值必须是cursor或arrayList
多表+筛选查询
可以从方法参数列表中给sql传入参数,从参数列表传入的要加上:(冒号)以示区分
@Query("select users.id,users.name from users,performs " +
"where users.id = performs.p_id and performs.score > :score and performs.assist > :assist")
// 多表+筛选查询
List<UserSimple> getUserWithLimits(int score, int assist);
可以看到,我的返回值List里的元素是UserSimple类型的,因为我们的sql是只查询了队员的号码和名字,而我们又在User类中固定了room能用的构造方法参数为3个(号码,名字,位置),这就出现了不匹配,room会编译时就拉闸的。
所以我又针对号码和名字建立了一个新的类UserSimple,由于这不是数据库中的表,所以不用也不能使用@Entity(用了就要addMigrate()),只需要注名属性和列名的对应关系就中。代码如下:
public class UserSimple {
@ColumnInfo(name = "id")
private String id;
@ColumnInfo(name = "name")
private String name;
//...get/set和toString()
}
插入
配上@Insert注解和冲突政策(为了省事儿,我直接替换)
@Insert(onConflict = OnConflictStrategy.REPLACE)
//插入有冲突,就直接替换
void insert(User[] users);
删除
由于我们定义外键时写了级联,所以这儿只配上一个@Delete就中
@Delete
int delete(User user);
//删除方法返回值必须是int或void
更新
配上@Update注解
@Update
int update(User user);
//更新方法返回值也必须是int或void,但不要这么更新,要么用query,指定sql语句,要么直接重新插入
//实际覆盖插入也可以实现更新
查询更新(推荐)
正如上面的注释所说,@Update并不能生成正确的sql语句(不信你单步调试看看),所以更新最好利用@Query注解,而后传入更新的sql就中了
@Query("update users set name=:name where id = :id")
void updateName(String name, long id);
使用
最好在Fragment中使用。另外,room默认是让数据库的IO操作必须放在子线程里,否则程序直接宕掉。
我为了方便,用了ButterKnife的注入和线程池的构造新线程的方法,但这里我们只看跟数据库有关的内容。
设置初始数据
private ArrayList<User> usersAll = new ArrayList<User>();
private ArrayList<UserPerforms> performsAll = new ArrayList<UserPerforms>()
private void setInfo() {
usersAll.add(new User(11, "Jason", "中锋"));
usersAll.add(new User(17, "Mike", "前腰"));
usersAll.add(new User(2, "Kane", "中卫"));
usersAll.add(new User(8, "Handerson", "后腰"));
usersAll.add(new User(1, "Frank", "门将"));
performsAll.add(new UserPerforms(11, 32, 15));
performsAll.add(new UserPerforms(1, 0, 0));
performsAll.add(new UserPerforms(17, 20, 23));
performsAll.add(new UserPerforms(8, 6, 10));
performsAll.add(new UserPerforms(2, 3, 4));
}
提取DAO
为了使用方便,我把获取DAO接口的方法提取一下:
public class CRUDUtil {
public static CRUDDAO getDao(Context context) {
return DatabaseCreator.getInstance(context).getDao();
}
}
可以看到,这儿就是调用了我们自己的DatabaseCreator类的抽象方法--getDao()
构造与DB交互类
构造一个模型和数据库交互的类,名曰MyModel,里面提供增查改删的方法,以供Fragment调用(为了方便,我就不用泛型了,直接重载)
public void insert(User[] data, Context context) {
CRUDUtil.getDao(context).insert(data);
}
public void insert(UserPerforms[] data, Context context) {
CRUDUtil.getDao(context).insert(data);
}
public void update(User data, Context context) {
CRUDUtil.getDao(context).updateName(data.getName(), data.getId());
}
public ArrayList<User> getAllUsers(Context context) {
return (ArrayList<User>) CRUDUtil.getDao(context).getAllUsers();
}
public ArrayList<UserPerforms> getAllPerforms(Context context) {
return (ArrayList<UserPerforms>) CRUDUtil.getDao(context).getAllPerforms();
}
public ArrayList<UserSimple> getUsersWithLimits(Context context, int score, int assist) {
return (ArrayList<UserSimple>) CRUDUtil.getDao(context).getUserWithLimits(score, assist);
}
public void delete(User u, Context context) {
CRUDUtil.getDao(context).delete(u);
}
测试
ThreadUtil.executeRunnable(new Runnable() {
@Override
public void run() { // 插入数据
nameModel.insert(usersAll.toArray(new User[1]), getContext());
nameModel.insert(performsAll.toArray(new UserPerforms[1]), getContext());
}
});
ThreadUtil.executeRunnable(new Runnable() {
@Override
public void run() { // 更新数据
usersAll.get(1).setName("szc");
nameModel.update(usersAll.get(1), getContext());
}
});
ThreadUtil.executeRunnable(new Runnable() {
@Override
public void run() { // 删除数据
for (int i = 0; i < usersAll.size(); i++) {
nameModel.delete(usersAll.get(i), getContext());
sleep(1000);
}
}
});
ThreadUtil.executeRunnable(new Runnable() {
@Override
public void run() {
usersShow = (ArrayList<User>) CRUDUtil.getDao(getContext()).getAllUsers();
// handler + message 遍历
}
});
ThreadUtil.executeRunnable(new Runnable() {
@Override
public void run() {
ArrayList<UserSimple> userSimples = (ArrayList<UserSimple>) CRUDUtil.getDao(getContext()).getUserWithLimits(10,10);
// handler + message 遍历
}
});
ThreadUtil是我自己封装的一个类,就是利用线程池执行子线程
利用RxJava构造响应式数据库
可以看到,虽然上面已经可以方便地对数据库进行CRUD操作,但是数据库的数据每变化一次,我们就要主动调用方法获取一次,稍显麻烦。为此,我们可以用CompositeDisposable对象充当观察者,进行监听。
获取Flowable对象
在CRUD接口中增加查询方法,返回值中List的元素变成Flowable对象
@Dao
public interface CRUDDAO {
....
@Query("select * from users")
Flowable<List<User>> getAllUsersFlowable();
....
}
更新交互类
在MyModel中加入相应方法
public Flowable<List<User>> getAllUsersFlowable(Context context) {
return CRUDUtil.getDao(context).getAllUsersFlowable();
}
创建并初始化CompositeDisposable
在Fragment中合适部分(我选的是onCreate()中),创建并初始化CompositeDisposable对象
private CompositeDisposable disposable = new CompositeDisposable();
disposable.add(nameModel.getAllUsersFlowable(getContext())
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Consumer<List<User>>() { // Flowable变成Disposable
@Override
public void accept(List<User> users) throws Exception {
Collections.sort(users);
traverseList((ArrayList) users); // 遍历
}
}));
这样,数据库User表一有动静(因为我们是获取的User表的Flowable),disposable对象就会观察到,调用getAllUsersFlowable()方法,不出意外的话,accept方法就会被调用,遍历就会被进行。
查看导出内容
最后,我们看一下数据库导出的内容(工程目录下schemas/包/json文件):
{
"formatVersion": 1,
"database": {
"version": 1,
"identityHash": "54e23e816a8145dfacd8880f58915619",
"entities": [
{
"tableName": "users",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER NOT NULL, `name` TEXT NOT NULL, `position` TEXT, `p_id` INTEGER, `score` INTEGER, `assist` INTEGER, PRIMARY KEY(`id`, `name`))",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "name",
"columnName": "name",
"affinity": "TEXT",
"notNull": true
},
{
"fieldPath": "position",
"columnName": "position",
"affinity": "TEXT",
"notNull": false
},
{
"fieldPath": "performs.p_id",
"columnName": "p_id",
"affinity": "INTEGER",
"notNull": false
},
{
"fieldPath": "performs.score",
"columnName": "score",
"affinity": "INTEGER",
"notNull": false
},
{
"fieldPath": "performs.assist",
"columnName": "assist",
"affinity": "INTEGER",
"notNull": false
}
],
"primaryKey": {
"columnNames": [
"id",
"name"
],
"autoGenerate": false
},
"indices": [
{
"name": "index_users_id",
"unique": true,
"columnNames": [
"id"
],
"createSql": "CREATE UNIQUE INDEX `index_users_id` ON `${TABLE_NAME}` (`id`)"
}
],
"foreignKeys": []
},
{
"tableName": "performs",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`p_id` INTEGER NOT NULL, `score` INTEGER NOT NULL, `assist` INTEGER NOT NULL, PRIMARY KEY(`p_id`), FOREIGN KEY(`p_id`) REFERENCES `users`(`id`) ON UPDATE CASCADE ON DELETE CASCADE )",
"fields": [
{
"fieldPath": "p_id",
"columnName": "p_id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "score",
"columnName": "score",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "assist",
"columnName": "assist",
"affinity": "INTEGER",
"notNull": true
}
],
"primaryKey": {
"columnNames": [
"p_id"
],
"autoGenerate": false
},
"indices": [],
"foreignKeys": [
{
"table": "users",
"onDelete": "CASCADE",
"onUpdate": "CASCADE",
"columns": [
"p_id"
],
"referencedColumns": [
"id"
]
}
]
}
],
"setupQueries": [
"CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)",
"INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, \"54e23e816a8145dfacd8880f58915619\")"
]
}
}