当前位置:   article > 正文

鸿蒙开发实例|对象关系映射数据库_@ohos.data.valuesbucket

@ohos.data.valuesbucket

 HarmonyOS应用数据管理不仅支持单设备的各种结构化数据的持久化,还支持跨设备之间数据的同步、共享及搜索功能,因此,开发者基于Harmony OS应用数据管理功能,能实现应用程序数据在不同终端设备之间的无缝衔接,从而保证用户在跨设备使用数据时所用数据的一致性。

HarmonyOS对象关系映射数据库是HarmonyOS中的本地数据库之一,是在SQLite数据库的基础上提供的一个抽象层。它屏蔽了底层SQLite数据库的SQL操作,将实例对象映射到关系上,并提供增、删、改、查等一系列面向对象的接口,使应用程序使用操作实例对象的语法来操作关系型数据库,因此,开发者可以不必编写那些复杂的SQL语句,从而既可以提升效率也能聚焦于业务开发。HarmonyOS中对象关系映射数据库的运作机制如图1所示。

从图1可以看出,对象关系映射数据库的3个主要组件包括:

(1) 数据库: 通过@Database注解,且继承自OrmDatabase的类,对应关系型数据库。

(2) 实体对象: 通过@Entity注解,且继承自OrmObject的类,对应关系型数据库中的表。

(3) 对象数据操作接口: 包括数据库操作的入口OrmContext 类和谓词接口(OrmPredicate)等。

图1 对象关系映射数据库的运作机制

创建Phone设备下的Java模板新项目。在进行开发之前,首先要进行build.gradle文件的配置,否则无法识别相关数据库的类的包。

一般情况下,使用的注解处理器的模块为com.huawei.ohos.hap模块,需要在模块的build.gradle文件的ohos节点中添加以下配置,代码如下:

  1. compileOptions{
  2. annotationEnabled true
  3. }

如果使用注解处理器的模块为com.huawei.ohos.library,则需要在模块的build.gradle文件dependencies节点中配置注解处理器,此外,还需要在本地的HUAWEI SDK中找到orm_annotations_java.jar、orm_annotations_processor_java.jar和javapoet_java.jar这3个jar包的对应目录,并将这3个jar包的路径导入,代码如下:

  1. dependencies {
  2.     compile files("orm_annotations_java.jar的路径","orm_annotations_processor_java.jar的路径","javapoet_java.jar的路径")
  3.     annotationProcessor files("orm_annotations_java.jar的路径","orm_annotations_processor_java.jar的路径","javapoet_java.jar的路径")
  4. }

如果使用注解处理器的模块为java-library,则还需要导入ohos.jar的路径,具体代码如下:

  1. dependencies {
  2.     compile files("ohos.jar的路径","orm_annotations_java.jar的路径","orm_annotations_processor_java.jar的路径","javapoet_java.jar的路径")
  3.     annotationProcessor files("orm_annotations_java.jar的路径","orm_annotations_processor_java.jar的路径","javapoet_java.jar的路径")
  4. }

配置完成之后,就可以开始新建数据库并对其进行操作了,具体步骤如下。

01、新建数据库及属性配置

创建数据库时,首先需要定义一个表示数据库的类,继承OrmDatabase,再通过@Database注解内的entities属性指定哪些数据模型类属于这个数据库,version属性指明数据库版本号。

在本例中,首先新建一个数据库类BookStore.java,选择entry→src→main→com.huawei.ormdb,右击选择并新建Class,命名为BookStore,如图2所示。

图2 新建一个数据库类

数据库类BookStore.java包含了User和Book两个表,版本号为1,将数据库类设置为虚类。具体代码如下:

  1. import ohos.data.orm.OrmDatabase;
  2. import ohos.data.orm.annotation.Database;
  3. @Database(entities = {User.class, Book.class}, version = 1)
  4. public abstract  class BookStore extends OrmDatabase {
  5. }

02、构造数据表

构造数据表,即创建数据库实体类并配置对应的属性(如对应表的主键、外键等)。可通过创建一个继承了OrmObject并用@Entity注解的类,获取数据库实体对象,也就是表的对象。需要注意的是,数据表必须与其所在的数据库在同一个模块中。以新建的User表为例,新建的User类与BookStore类位于同一模块下,相应的目录如图3所示。

图3 BookStore数据库与User表位于同一模块

具体的构建过程代码如下:

  1. //新建User表
  2. import ohos.data.orm.OrmObject;
  3. import ohos.data.orm.annotation.Entity;
  4. import ohos.data.orm.annotation.Index;
  5. import ohos.data.orm.annotation.PrimaryKey;
  6. @Entity(tableName = "user", ignoredColumns = {"ignoreColumn1", "ignoreColumn2"},
  7. indices = {@Index(value = {"firstName", "lastName"}, name = "name_index", unique = true)})
  8. public class User extends OrmObject {
  9. //此处将userId设为了自增的主键。注意只有在数据类型为包装类型时,自增主键才能生效。
  10. @PrimaryKey(autoGenerate = true)
  11. private Integer userId;
  12. private String firstName;
  13. private String lastName;
  14. private int age;
  15. private double balance;
  16. private int ignoreColumn1;
  17. private int ignoreColumn2;
  18. //设置字段的getter和setter方法,此处仅给出示例,读者可以根据所设置的属性自行补全方法
  19. public void setBalance(double balance) {
  20. this.balance = balance;
  21. }
  22. ...
  23. public Integer getUserId() {
  24. return userId;
  25. }
  26. ...
  27. }

 在新建的User类中进行属性配置,tableName = "user"即在对应数据库内的表名为user,indices为属性列表,@Index注解的内容对应数据表索引的属性,本例中indices为firstName和lastName两个字段建立了复合索引,索引名为name_index,并且索引值是唯一的。ignoreColumns 表示该字段不需要添加到user 表的属性中,即类中定义的ignoreColumn1和ignoreColumn2不属于user表的属性。被@PrimaryKey注解的变量对应数据表的主键,一个表里只能有一个主键,在本例的user表中,将userId设为自增的主键。

数据库内还包含了Book表,Book表的构建和对其操作的实现过程同User表相同,具此处不再赘述其体实现。

03、创建数据库

在MainAbilitySlice的onStart()方法中完成数据库的创建,使用对象数据操作接口OrmContext创建数据库,实现代码如下:

  1. public void onStart(Intent intent) {
  2. super.onStart(intent);
  3. super.setUIContent(ResourceTable.Layout_ability_main);
  4. //使用对象数据操作接口OrmContext创建数据库
  5. DatabaseHelper helper = new DatabaseHelper(this);
  6. OrmContext context = helper.getOrmContext("BookStore", "BookStore.db", BookStore.class);
  7. //对数据库进行操作
  8. //增加数据
  9. ...
  10. //查询数据
  11. ...
  12. //修改数据
  13. ...
  14. //删除数据
  15. ...
  16. }

与构建关系型数据库不同的是,此处new DatabaseHelper(context)方法中context的入参类型为ohos.app.Context,必须直接传入slice而不能使用slice.getContext()获取context,否则会出现找不到类的报错。到这里,就成功创建了一个别名为BookStore且数据库文件名为BookStore.db的数据库。如果数据库已经存在,则执行上述代码并不会重复建库。通过context.getDatabaseDir()可以获取创建的数据库文件所在的目录。

04、数据操作

对象数据操作接口OrmContext提供了对数据库进行增、删、改、查的一系列方法,接下来进行详细介绍。

(1)增加数据 。

例如,在名为user 的表中,新建一个User 对象并设置其属性。OrmContext提供insert()方法将对象插入数据库。执行完insert()方法后,数据被保存在内存中,只有在flush()被调用后才会将数据持久化到数据库中。以HiLog形式显示插入数据后的user表,如图4所示。此处涉及对数据库的查询操作,将在稍后进行详细讲解。具体代码如下:

  1. //增加数据
  2. //新建User对象
  3. User user = new User();
  4. //设置对象属性
  5. user.setUserId(0);
  6. user.setFirstName("aa");
  7. user.setLastName("AA");
  8. user.setAge(22);
  9. user.setBalance(120.51);
  10. //再新建两个User对象,设置属性后将其持久化到数据库中,此处读者可以自行添加代码
  11. ...
  12. //将新建对象插入并持久化到数据库中
  13. boolean isSuccessed = context.insert(user);
  14. isSuccessed = context.flush();
  15. //以HiLog形式显示新增数据后的user表,以userID升序显示
  16. HiLogLabel logLabel = new HiLogLabel(HiLog.LOG_APP,0,"OrmDB");
  17. String s = new String("userID");
  18. s = s.concat(" firstName");
  19. s = s.concat(" lastName");
  20. s = s.concat(" age");
  21. s = s.concat(" balance");
  22. HiLog.fatal(logLabel,"显示插入数据后的user表:");
  23. HiLog.fatal(logLabel,s);
  24. //查询数据
  25. OrmPredicates query = context.where(User.class).orderByAsc("userID");
  26. List<User> users = context.query(query);
  27. int i = 0;
  28. while (i != users.size()) {
  29. int id = users.get(i).getUserId();
  30. String s1 = "\t" + id + "\t\t";
  31. s1 = s1.concat(users.get(i).getFirstName() + "\t");
  32. s1 = s1.concat("\t\t" + users.get(i).getLastName());
  33. s1 = s1.concat("\t " + users.get(i).getAge() + "\t");
  34. s1 = s1.concat(" " + users.get(i).getBalance());
  35. HiLog.fatal(logLabel, s1);
  36. i = i + 1;
  37. }

(2) 查询数据。

OrmContext提供query()方法查询满足指定条件的对象实例。例如,在user表中查询age=22的对象,在查询之前,依旧需要先设置谓词query1,利用query()方法查找user表中满足query1 的数据,得到一个User的列表user1,遍历user1,获取user1中各个对象的属性值,实现代码如下:

  1. //查询age=22的数据并显示
  2. //设置谓词
  3. OrmPredicates query1 = context.where(User.class).equalTo("age", "22");
  4. //查询
  5. List<User> users1 = context.query(query1);
  6. //将查询结果以HiLog形式显示
  7. HiLogLabel logLabel1 = new HiLogLabel(HiLog.LOG_APP,0,"OrmDB");
  8. HiLog.fatal(logLabel1,"显示user表中age值为22的数据:");
  9. HiLog.fatal(logLabel1,s);
  10. i = 0;
  11. while (i != users1.size()){
  12. int id = users1.get(i).getUserId();
  13. String s1 = "\t" + id +"\t\t";
  14. s1 = s1.concat(users1.get(i).getFirstName()+"\t");
  15. s1 = s1.concat("\t\t"+users1.get(i).getLastName());
  16. s1 = s1.concat("\t "+users1.get(i).getAge()+"\t");
  17. s1 = s1.concat(" "+users1.get(i).getBalance());
  18. HiLog.fatal(logLabel1,s1);
  19. i = i + 1;
  20. }

 查询结果如图5所示。

图5 查询user表中age=22的数据

(3) 修改数据。

修改数据包括两种方式,一种是通过直接传入OrmObject对象的接口来更新数据,需要先从表中查到需要更新的结果对象列表,然后修改选定对象的值,再调用更新接口将数据持久化到数据库中。例如,将user表中age=39的数据的firstName更新为sd,需要先查找user表中对应的数据,得到一个结果列表,然后选择列表中需要更新的User对象,设置更新值,并调用update接口传入被更新的User对象。最后调用flush接口将数据持久化到数据库中。具体代码如下:

  1. //更新数据
  2. OrmPredicates predicates = context.where(User.class);
  3. predicates.equalTo("age", 39);
  4. //获取满足条件的数据集
  5. List<User> users2 = context.query(predicates);
  6. //选定要更新的数据
  7. User userUD = users2.get(0);
  8. //设置更新
  9. userUD.setFirstName("sd");
  10. context.update(userUD);
  11. context.flush();
  12. HiLog.fatal(logLabel, "显示更新数据后的user表:");
  13. HiLog.fatal(logLabel, s);
  14. query = context.where(User.class).orderByAsc("userID");
  15. users = context.query(query);
  16. i = 0;
  17. while (i != users.size()) {
  18. int id = users.get(i).getUserId();
  19. String s1 = "\t" + id + "\t\t";
  20. s1 = s1.concat(users.get(i).getFirstName() + "\t");
  21. s1 = s1.concat("\t\t" + users.get(i).getLastName());
  22. s1 = s1.concat("\t " + users.get(i).getAge() + "\t");
  23. s1 = s1.concat(" " + users.get(i).getBalance());
  24. HiLog.fatal(logLabel, s1);
  25. i = i + 1;
  26. }

 另一种方式,可以通过传入谓词的接口来更新和删除数据,方法与OrmObject对象的接口类似,只是无须flush就可以将数据持久化到数据库中。通过这种方式将user表中age=39的数据的firstName更新为sd,具体代码如下:

  1. //使用valuesBucket来改变数据
  2. ValuesBucket valuesBucket = new ValuesBucket();
  3. valuesBucket.putString("firstName", "sd");
  4. OrmPredicates update = context.where(User.class).equalTo("age", 39);
  5. context.update(update, valuesBucket);

分别利用两种方式进行数据更新,更新后的结果如图6所示。

(4) 删除数据。

同修改数据一样,也包括两种方式。区别只是删除数据不需要更新对象的值。这里通过直接传入OrmObject对象的接口删除数据的方式删除满足firstName="aa"的第一个User对象,并用flush接口将数据持久化到数据库中,代码如下

  1. //删除满足firstName = "aa"的第一个对象
  2. OrmPredicates del = context.where(User.class).equalTo("firstName", "aa");
  3. List<User> usersdel = context.query(del);
  4. User userd = usersdel.get(0);
  5. context.delete(userd);
  6. context.flush();
  7. HiLog.fatal(logLabel, "显示删除数据后的user表:");
  8. HiLog.fatal(logLabel, s);
  9. query = context.where(User.class).orderByAsc("userID");
  10. users = context.query(query);
  11. i = 0;
  12. while (i != users.size()) {
  13. int id = users.get(i).getUserId();
  14. String s1 = "\t" + id + "\t\t";
  15. s1 = s1.concat(users.get(i).getFirstName() + "\t");
  16. s1 = s1.concat("\t\t" + users.get(i).getLastName());
  17. s1 = s1.concat("\t " + users.get(i).getAge() + "\t");
  18. s1 = s1.concat(" " + users.get(i).getBalance());
  19. HiLog.fatal(logLabel, s1);
  20. i = i + 1;
  21. }

显示删除数据后的表,如图7所示。

 图7 删除数据后的结果显示

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/你好赵伟/article/detail/236810
推荐阅读
相关标签
  

闽ICP备14008679号