- 实体 - 表 对应 主键 外键(一对一,一对多)
- 获取Database
- dao操作
- 加密
- 数据库升级
- LiveData的支持
框架图
Room:
GreenDao:
实体 - 表
- Room
1、创建一个实体ProductEntity.java 使用@Entity注解标注:@Entity可以指定表名tableName,添加索引,外键等。
2、如果不指定tableName,默认是用类名作为数据库的表名。
3、主键使用PrimaryKey注解,可以设置autoGenerate = true表示主键自增。
4、其他属性默认都会作为表中的字段,如果不想要该字段可以使用@Ignore注解,表示忽略此属性。
5、你可以指定对象之间的关联。虽然大多数ORM库允许entity对象相互引用,但是Room明确禁止了这种行为。
虽然不可以使用直接的关联,Room仍然允许你定义entity之间的外键(Foreign Key)约束。
@Entity(tableName = "products",
indices = {@Index(value = {"id","name"},unique = true)})
public class ProductEntity implements Product{
@PrimaryKey(autoGenerate = true)
private Long id;
private String name;
private String description;
private double price;
@Ignore
private String testField;
}
- GreenDao
1、创建一个实体类ProductEntity.java,使用@Entity注解标注:@Entity可以指定表名nameInDb=“”,索引等
2、如果不指定nameInDb,默认使用类名作为表名。
3、主键使用@Id注解标注,可以设置autoincrement = true让主键自增,注意:这里主键类型必须是Long,因为只有id=null时自增才会生效。
4、其他属性默认都会作为表中的字段,如果不需要时使用Transient注解忽略此属性。
5、支持对象之间的关联(一对一,一对多等等)
@Entity
public class ProductEntity implements Product {
@Id(autoincrement = true)
private Long id; //这个逼玩意必须是Long,因为在数据库中只有id=null时才会自增。
private String name;
private String description;
private double price;
@Transient //忽略此属性
private String testColumn;
}
Database
Room:
1、使用Room时需要创建自己的AppDatabasele抽象类,并且继承自RoomDatabase。
2、使用@Database注解标注,其中可以指定entitys,将这个库中所有的表对应的实体放在entitys中,version字段可以指定数据库版本。
3、Dao对象需要开发者编写各entity对应的Dao接口,并用@Dao注解标注,接口中提供需要的方法声明。Room会自动生成对应的实现类,并实现所有的方法。
@Dao
public interface ProductDao {
@Query("select * from products")
List<ProductEntity> queryProducts();
@Query("select * from products where id = :productId")
ProductEntity getProductById(long productId);
@Insert(onConflict = OnConflictStrategy.REPLACE)
void insertAll(ProductEntity... entity);
@Update(onConflict = OnConflictStrategy.REPLACE)
void updateAll(ProductEntity... entity);
@Delete
void deleteAll(ProductEntity... entity);
}
4、在AppDatabase中提供对应实体的Dao的方法声明。这样就可以通过APPDatabase实例获取到指定的dao对应从而操作表了。
说明:Room会自动生成该类的实现类。使用时直接调用即可。代码如下:
@Database(entities = {ProductEntity.class, CommentEntity.class},version = 1)
@TypeConverters(DateConverter.class)
public abstract class AppDatabase extends RoomDatabase {
public static String DATABASE_NAME = "room-sample-db";
public abstract ProductDao productDao();
public abstract CommentDao commentDao();
}
5、获取数据库实例
使用Room提供的静态方法创建数据库。代码如下:
appDatabase = Room.databaseBuilder(context.getApplicationContext(),
AppDatabase.class,DATABASE_NAME)
.addMigrations(MIGRATION_1_2)
.build();
GreenDao
1、使用GreenDao时会自动生成DaoMaster,DaoSession,XXXEntityDao三个类。
DaoMaster
- 是GreenDao的入口,也是greenDao顶级对象,对于一个指定的表单持有数据库对象(SQLite数据库)并且能够管理DAO类- 能够创建表和删除表
- 其内部类OpenHelper 与DevOpenHelper是创建SQlite数据库的SQLiteOpenHelper的具体实现
DaoSession
- 对于一个指定的表单可以管理所有的Dao 对象。
- 也能够对实体类执行 insert ,load,update,refresh.delete操作。
- DaoSession也能跟踪 identity scope:即session查询后的实体会存在缓存中,并给该实体生成一个flag来追踪该实体,下次再次查询时会直接从缓存中取出来而不是从数据库中取出来
Daos
- 能够持久访问和查询实体类
- 比起DaoSession有更多的持久化方法 count, loadAll,insertInt等等;
2、获取数据库实例
1)、首先需要获取SQLiteOpenHelper对象。可以通过DaoMaster中的DevOpenHelper来实例化,也可以自定义类实现DaoMaster中的OpenHelper.
SQLiteOpenHelper oh = new DaoMaster.DevOpenHelper(context,DB_NAME,null);
2)、获取DaoSession: 通过DaoMaster的newSession()获取。
mDatabase = oh.getWritableDb();
daoSession = new DaoMaster(mDatabase).newSession();
3)、获取Dao对象:通过DaoSession获取
daoSession.getProductEntityDao();
注:
1、Room将生成Dao的控制权交由开发者,开发者决定需要那些dao。这样可以避免将所有的Entity都生成对应的Dao。更加灵活可控,也可以减少代码量。
GreenDao的工作方式是你只需要给我一个Entity,它就会生成所有东西,使用时只需要取对应的对象去操作即可。灵活性不如Room。
2、GreenDao提供了DaoSession会话,缓存层。可以很大程度来提高查询效率。Room貌似没有这样的缓存。
操作Sql(增删改查)
insert,update,delete
Room:使用@Insert,@Update,@Delete注解,并且可以指定相应的策略(replace,rollback,fail等)
GreenDao:直接生成对应的方法。没有提供相应的策略。但是提供了执行sql的方法。
query
Room:使用@Query注解标注,直接传sql
GreenDao:以对象的方式操作。可以使用QueryBuilder类执行各种复杂的查询。
两者做法各有千秋,就看使用者的习惯。就我个人而言还更喜欢使用纯sql,可读性高。
加密
Room不支持SqlCipher加密,不过后续SqlCipher应该会提供针对Room的加密方式,这只是时间的问题。
GreenDao集成了SqlCipher加密。
数据库升级
Room
生成AppDatabase实例时可以通过addMigrations()添加对应的Migration来支持迁移操作。代码如下:
appDatabase = Room.databaseBuilder(context.getApplicationContext(),
AppDatabase.class,DATABASE_NAME)
.addMigrations(MIGRATION_1_2)
.build();
private static Migration MIGRATION_1_2 = new Migration(1,2) {
@Override
public void migrate(SupportSQLiteDatabase database) {
//// TODO: 2017/6/12
}
};
GreenDao
通过重写DaoMaster.OpenHelper类实现迁移操作:
public static class AppSQLiteOpenHelper extends DaoMaster.OpenHelper{
public AppSQLiteOpenHelper(Context context, String name) {
super(context, name);
}
public AppSQLiteOpenHelper(Context context, String name, SQLiteDatabase.CursorFactory factory) {
super(context, name, factory);
}
@Override
public void onUpgrade(Database db, int oldVersion, int newVersion) {
//数据库迁移操作
MigrationHelper.getInstance().migrate(db,ProductEntityDao.class, CommentEntityDao.class);
}
}
Room添加对liveData的支持
GreenDao不支持LiveData,需要自己实现。