Android Jetpack组件之数据库Room详解(一)

本文涉及Library的版本如下:

  • androidx.room:room-runtime:2.1.0-alpha03
  • androidx.room:room-compiler:2.1.0-alpha03(注解编译器)
  • androidx.room:room-rxjava2:2.1.0-alpha03

Room的介绍

Room为了SQLite提供了一个抽象层,对Android SQLite进行了封装 , 更加方便地进行数据库访问。

Room主要有3个组件:

  • Database: 数据库的持有者。
  • Entity: 代表数据库一张表
  • DAO: 数据访问对象

Room架构图如下:

room_architecture

从上图可以知道,应用层需要定义有哪些表; 定义数据访问对象; 定义数据库,比如说数据库名称,有哪些些表等。下面结合一些简单的代码例子:

// 定义User表, 用Entity注解来表示这个类是一个数据库表
@Entity
public class User {
	// 使用ColumnInfo注解定义一个字段名, 不用注解默认去变量名
    @ColumnInfo(name = "first_name") 
    public String firstName;

    @PrimaryKey     // 主键
    public int id;
}
// 定义数据访问对象的接口
@Dao
public interface UserDao {  // 定义成接口
    // Query注解定义查询, 参数是sql语句
    @Query("SELECT * FROM user")  
    List<User> getAll();
    
    // 根据id查询user, :id这里意思是引用findById方法里参数id。是room定义固定写法:冒号+参数名称
    @Query("SELECT * FROM user WHERE id = :id") 
    User findById(String id);
	//Update注解定义更新User
    @Update
    void udapte(User user);
	//Insert注解定义插入User
    @Insert
    void insert(User user);
	//Delete注解定义删除User
    @Delete
    void delete(User user);
}
//定义AppDatabase类,需要是abstract的,继承RoomDatabase
//使用Database注解,定义entities类, entities参数是一个class[],version是数据库的版本号
@Database(entities = {User.class, Favorite.class}, version = 1)
public abstract class AppDatabase extends RoomDatabase {
    public abstract UserDao userDao();
    public abstract FavoriteDao favoriteDao();
    
    private static final String DB_NAME = "room_demo";
    private static volatile AppDatabase sInstance;

    //通常定义一个单例持有AppDatabase引用
    //DB_NAME是数据库文件名称
    public static AppDatabase getInstance(Application app){
        if(sInstance == null){
            synchronized (AppDatabase.class){
                if(sInstance == null){
                    sInstance = Room.databaseBuilder(app,
                            AppDatabase.class, DB_NAME).build();
                }
            }
        }
        return sInstance;
    }
}

//操作数据库增删查
mAppDatabase = AppDatabase.getInstance(this.getApplication());
mAppDatabase.userDao().delete(user);
mAppDatabase.userDao().insert(user);
mAppDatabase.userDao().getAll();

数据库迁移

通过RoomDatabase.Builder.addMigrations()添加Migration去实现。代码示例如下:

    static final Migration MIGRATION_1_2 = new Migration(1, 2) {
        @Override
        public void migrate(SupportSQLiteDatabase database) {
//            database.execSQL("");执行sql语句
        }
    };

    static final Migration MIGRATION_2_3 = new Migration(2, 3) {
        @Override
        public void migrate(SupportSQLiteDatabase database) {
//            database.execSQL("");执行sql语句
        }
    };
    
Room.databaseBuilder(app,AppDatabase.class, DB_NAME)
    .addMigrations(MIGRATION_1_2, MIGRATION_2_3)
    .build();    

支持RxJava、LiveData的扩展

在定义Dao接口时,可以直接LiveData和Flowable来作为返回数据类型。

@Dao
public interface UserDao {
    @Query("SELECT * FROM user")
    LiveData<List<User>> getUsersLiveData();

    @Query("SELECT * FROM user")
    Flowable<List<User>> getUsersFlowable();
}    

//LiveData使用例子
public class RoomActivity extends AppCompatActivity {
    
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mRoomModel = ViewModelProviders.of(this).get(RoomModel.class);
        mRoomModel.getUsersLiveData().observe(this, new Observer<List<User>>() {
            @Override
            public void onChanged(List<User> users) {
                adapter.setData(users);//更新UI
            }
        });
    }
}    

public class RoomModel extends AndroidViewModel {
    private final AppDatabase mAppDatabase;
    public RoomModel(@NonNull Application application) {
        super(application);
        mAppDatabase = AppDatabase.getInstance(this.getApplication());
    }

    public LiveData<List<User>> getUsersLiveData() {
        return mAppDatabase.userDao().getUsersLiveData();
    }
}    

//RxJava使用例子
mAppDatabase.userDao().getUsersFlowable()
    .subscribeOn(Schedulers.io())
    .observeOn(AndroidSchedulers.mainThread())
    .subscribe(new Consumer<List<User>>() {
        @Override
        public void accept(List<User> users) throws Exception {
            adapter.setData(users);//更新UI
        }
    });

对RxJava不熟悉自己去了解, 对 LiveData不了解可以我之前的文章LiveData

一些遗留的问题

  • 利用注解Database并且继承RoomDatabase,到底是怎么生存数据库

  • 数据访问只需要定义一个DAO接口, Dao接口真正实现是怎样的,增删改查是如何实现的

  • 是如何扩展RxJava和LiveData, 并且如何能监听Flowable和LiveData数据的变更

参考:*

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Android Jetpack是Google提供的一套用于加速Android应用开发的工具包,其中包括了许多架构组件,其中之一就是ViewModel。 ViewModel是一种设计模式,用于保存和管理与UI相关的数据。在传统的Android开发中,当屏幕旋转或者因为其他原因导致Activity或Fragment重建时,之前保存的临时数据就会丢失。而ViewModel的出现解决了这个问题。 ViewModel的主要作用是将数据与UI组件分离。它的工作方式是创建一个ViewModel类,并在其中保存需要与UI组件交互的数据。这样,当屏幕旋转或重建时,ViewModel实例不会销毁,数据也会得到保留。然后,在Activity或Fragment中,通过获取ViewModel实例,可以轻松地访问这些数据。 使用ViewModel的好处有很多。首先,它可以避免内存泄漏,因为ViewModel的生命周期与Activity或Fragment无关。其次,它可以节省资源,因为当Activity或Fragment销毁时,ViewModel实例可以被系统缓存起来,下次再创建时可以直接返回该实例。另外,由于ViewModel保存了与UI相关的数据,可以减少因为屏幕旋转导致的数据重复加载的问题。 在使用ViewModel时,你可以选择使用Android Jetpack中的其他架构组件来进一步提高开发效率,比如通过LiveData实现数据的观察和通知,或者通过DataBinding来实现UI与数据的自动绑定。 总之,ViewModel是Android Jetpack中非常重要的一个架构组件,它的出现实现了数据与UI的解耦,提高了开发效率,并且解决了数据丢失的问题。希望通过这篇文档的详解,你对ViewModel有了更深入的理解。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值