Room VS GreenDao

0.144字数 1388阅读 6243
  • 实体 - 表 对应 主键 外键(一对一,一对多)
  • 获取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,需要自己实现。

推荐阅读更多精彩内容