Android本地数据存储Room实践和优化技巧

作者:顽石九变 时间:2023-11-04 09:01:30 

Room在SQLite基础上做了ORM封装,使用起来类似JPA,不需要写太多的sql。

导入依赖

//room
def room_version="2.4.2"
implementation "androidx.room:room-runtime:$room_version"
annotationProcessor "androidx.room:room-compiler:$room_version"
//implementation "androidx.room:room-rxjava2:$room_version"
//implementation "androidx.room:room-rxjava3:$room_version"
//implementation "androidx.room:room-guava:$room_version"
//testImplementation "androidx.room:room-testing:$room_version"
//implementation "androidx.room:room-paging:2.5.0-alpha01"

关键注解说明

1、@Database:Room数据库对象。该类需要继承自RoomDatabase,通过Room.databaseBuilder()结合单例设计模式,完成数据库的创建工作。我们创建的Dao对象,在这里以抽象方法的形式返回,只需一行代码即可。

  • entities:指定该数据库有哪些表

  • version:指定数据库版本号,后续数据库的升级正是依据版本号来判断的

2、@Entity:该类与Room中表关联起来。tableName属性可以为该表设置名字,如果不设置,则表名与类名相同。

3、@PrimaryKey:用于指定该字段作为表的主键。

4、@ColumnInfo:设置该字段存储在数据库表中的名字并指定字段的类型;默认字段名和属性名一样

5、@Ignore:忽略该字段

一、使用步骤

1、创建实体类,对应数据库中一张表,使用注解@Entity

2、创建Dao接口类,用于操作数据,使用注解@Dao;不需要实现,在编译的时候,框架会自动生成实现类

3、创建数据库对象Database,继承RoomDatabase,使用单例模式返回实例

4、在Activity中使用,Room数据操作必须在异步线程中执行,所以在Activity中使用线程池执行,或者使用RxJava切换线程

使用代码示例

1、创建实体类,对应数据库中一张表,使用注解@Entity

@Entity
public class Person {
   // 主键,自增长
   @PrimaryKey(autoGenerate = true)
   private int id;
   private String name;
   private String sex;
   private int age;
}

2、创建Dao接口类,用于操作数据,使用注解@Dao;不需要实现,在编译的时候,框架会自动生成实现类

@Dao
public interface PersonDao {
   // 插入
   @Insert
   void insertPersons(Person... persons);
   // 修改
   @Update
   void updatePersons(Person... persons);
   // 删除所有
   @Query("delete from Person")
   void deleteAllPersons();
   // 删除指定实体
   @Delete
   void deletePersons(Person... persons);
   // 根据id删除
   @Query("delete from Person where id in (:ids)")
   void deleteByIds(int ...ids);
   // 根据id查询
   @Query("select * from Person where id in (:ids)")
   List<Person> selectByIds(int ...ids);
   // 查询所有
   @Query("select * from Person order by id desc")
   List<Person> selectAllPersons();
}

3、创建数据库对象Database,继承RoomDatabase,使用单例模式返回实例

@Database(entities = {Person.class}, version = 1)
public abstract class AppDatabase extends RoomDatabase {
   public abstract PersonDao personDao();
   private volatile static AppDatabase instance;
   public static AppDatabase getInstance(Context context){
       if (instance == null) {
           synchronized (DBHelper.class) {
               if (instance == null) {
                   instance = Room.databaseBuilder(context, AppDatabase.class, "person.db").build();
               }
           }
       }
       return instance;
   }
}

4、在Activity中使用

Room数据操作必须在异步线程中执行,所以在Activity中使用线程池执行

ExecutorService pool = Executors.newCachedThreadPool();
// 插入数据
public void insertRoom(View view) {
   AppDatabase db = AppDatabase.getInstance(getApplicationContext());
   pool.execute(() -> {
       PersonDao dao = db.personDao();
       Person p1 = new Person("用户1", "男", 18);
       Person p2 = new Person("用户2", "男", 28);
       Person p3 = new Person("用户3", "男", 38);
       dao.insertPersons(p1, p2, p3);
   });
}
// 查询数据
public void queryRoom(View view) {
   AppDatabase db = AppDatabase.getInstance(getApplicationContext());
   pool.execute(() -> {
       PersonDao dao = db.personDao();
       List<Person> list = dao.selectAllPersons();
       list.forEach(p-> Log.d("test", p.toString()));
   });
}
// 根据id查询
public void queryRoomById(View view) {
   AppDatabase db = AppDatabase.getInstance(getApplicationContext());
   pool.execute(() -> {
       PersonDao dao = db.personDao();
       List<Person> list = dao.selectByIds(3,4);
       list.forEach(p-> Log.d("test", p.toString()));

});
}
// 删除
public void deleteRoom(View view) {
   AppDatabase db = AppDatabase.getInstance(getApplicationContext());
   pool.execute(() -> {
       PersonDao dao = db.personDao();
       dao.deleteByIds(1,2);
   });
}

二、类型转换器

SQLite支持null,integer,real,text,blob五种数据类型,实际上SQLite也接受varchar,char,decimal等数据类型,只不过在运算中或保存时会转换成对应的5种数据类型,因此,可以将各种类型数据保存到任何字段中。

除了上述基本类型外,其他如Date、BigDecimal、或Json对象等如何存储呢?

Room给我们提供的非常方便的类型转换器功能。

  • @TypeConverter,定义类型转换静态方法

  • @TypeConverters,定义包含一组转换方法的class类

1、创建类型转换类型,如,Date和Long互转

使用注解@TypeConverter声明具体的转换方法,每个方法必须包含一个参数,以及必须有返回值。

public class DateConverter {
   @TypeConverter
   public static Date toDate(Long dateLong) {
       return dateLong == null ? null : new Date(dateLong);
   }
   @TypeConverter
   public static Long fromDate(Date date) {
       return date == null ? null : date.getTime();
   }
}

2、将创建好的转换器类,在entity上使用

使用注解@TypeConverters({DateConverter.class}),那么实体类中的所有的Date属性都会被转换成Long存储,查询取出的时候,会自动从Long转换成Date显示。

注意:@TypeConverters放在元素属性、Class、Dao、Database上面

  • 放在元素属性,只对改属性有效

  • 放在实体Class上,对class中所有元素有效

  • 放在Dao上,对Dao的所有方法有效

  • 放在Database,对Database的所有实体和所有Dao都有效

为避免出现混乱,通常建议只在Entity或属性上定义转换器

@Entity
@TypeConverters({DateConverter.class})
public class BsGoods {
   private static final long serialVersionUID = 1122172437556010779L;
   // 主键
   @PrimaryKey
   private Long id;
   private Date createdDate;
   private Date updatedDate;
   ...
}

其他类型转换示例,BigDecimal转String。

如果是JavaBean等复杂对象,可以转换成Json字符串存储。

public class BigDecimalConverter {
   @TypeConverter
   public static String toStr(BigDecimal decimal) {
       return decimal == null ? null : decimal.toString();
   }
   @TypeConverter
   public static BigDecimal toDecimal(String str) {
       return str == null ? null : new BigDecimal(str);
   }
}

三、结合RxJava

在Activity中使用,并且更新界面UI元素

Android的界面UI元素更新,必须在主线程中执行,但是Room的数据查询,又只能使用异常线程处理。那么如何将查询到数据,更新到页面控件上面呢?

这里可以结合RxJava实现流式操作,线下切换!

示例代码,查询所有商品数据,显示在页面控件上面,控件使用的是自定义的TableView,暂不展开,这里只显示数据查询以及显示。

1、在Database类中定义查询方法,传入回调函数

public void selectAll(Consumer<List<BsGoods>> fun) {
   BsGoodsDao dao = bsGoodsDao();
   Observable.just("select")
           .map(s -> dao.selectAll())
           .subscribeOn(Schedulers.io())// 给上面的操作分配异步线程
           .observeOn(AndroidSchedulers.mainThread())// 给终点分配安卓主线程
           .subscribe(new Observer<List<BsGoods>>() {
               @Override
               public void onSubscribe(@NonNull Disposable d) {
               }
               @Override
               public void onNext(@NonNull List<BsGoods> bsGoods) {
                   fun.accept(bsGoods);
               }
               @Override
               public void onError(@NonNull Throwable e) {
               }
               @Override
               public void onComplete() {
               }
           });
}

2、在Activity中使用,传入回调函数更新界面UI

private void initializeTableViewLocal() {
   BsGoodsDatabase db = BsGoodsDatabase.getInstance(getContext());
   db.selectAll(list -> {
       GoodsTableViewModel tableViewModel = new GoodsTableViewModel(list);
       TableViewAdapter tableViewAdapter = new TableViewAdapter(tableViewModel);
       mTableView.setAdapter(tableViewAdapter);
       mTableView.setTableViewListener(new TableViewListener(mTableView));
       tableViewAdapter.setAllItems(tableViewModel.getColumnHeaderList(), tableViewModel
               .getRowHeaderList(), tableViewModel.getCellList());
   });
}

来源:https://blog.csdn.net/wlddhj/article/details/128286193

标签:Android,本地,存储,Room
0
投稿

猜你喜欢

  • Java synchronized关键_动力节点Java学院整理

    2023-11-10 11:08:53
  • Android如何使用Bmob后端云实现失物招领功能

    2023-10-15 07:28:05
  • C#使用FtpWebRequest与FtpWebResponse完成FTP操作

    2021-08-24 04:58:26
  • Java中的这些骚操作你不能不知道!!!

    2022-07-08 12:28:14
  • 总结一次C++ 程序优化历程

    2023-11-02 22:38:30
  • C#将隐私信息(银行账户,身份证号码)中间部分特殊字符替换成*

    2022-02-18 05:39:36
  • B/S与C/S架构的区别介绍

    2023-07-02 13:24:33
  • Android编程实现canvas绘制饼状统计图功能示例【自动适应条目数量与大小】

    2023-05-10 23:27:41
  • SpringBoot整合Redis之编写RedisConfig

    2023-08-29 02:35:57
  • SpringBoot集成Beetl后统一处理页面异常的方法

    2023-11-10 19:57:55
  • 详解spring cloud ouath2中的资源服务器

    2022-09-24 15:36:43
  • 如何利用泛型封装通用的service层

    2023-05-15 04:55:43
  • Spring创建Bean的6种方式详解

    2022-07-18 19:10:13
  • Android音视频开发之MediaExtactor使用教程

    2023-03-20 19:25:11
  • Android实战教程第一篇之最简单的计算器

    2023-03-12 17:57:54
  • android Retrofit2网络请求封装介绍

    2022-02-09 18:46:49
  • Java编程实现非对称加密的方法详解

    2023-08-24 01:21:26
  • AsyncTask官方文档教程整理

    2023-07-31 20:25:08
  • Android Drawable及其相关类的使用

    2023-10-18 09:32:38
  • c#(Socket)同步套接字代码示例

    2023-12-16 04:03:13
  • asp之家 软件编程 m.aspxhome.com