【Android -- 开源库】数据库 Realm 的基本使用

news/2024/7/10 19:58:26 标签: 数据库, android, 开源

简介

Realm 是一个 MVCC (多版本并发控制)数据库,由Y Combinator公司在2014年7月发布一款支持运行在手机、平板和可穿戴设备上的嵌入式数据库,目标是取代 SQLite。Realm 本质上是一个嵌入式数据库,他并不是基于 SQLite 所构建的。它拥有自己的数据库存储引擎,可以高效且快速地完成数据库的构建操作。和 SQLite 不同,它允许你在持久层直接和数据对象工作。在它之上是一个函数式风格的查询 API,众多的努力让它比传统的SQLite 操作更快 。

GitHub 地址:realm-java

优点

  • 易用
    Ream 不是在SQLite基础上的ORM,它有自己的数据查询引擎。并且十分容易使用。

  • 快速
    由于它是完全重新开始开发的数据库实现,所以它比任何的ORM速度都快很多,甚至比SLite速度都要快。

  • 跨平台
    Realm 支持 iOS & OS X (Objective‑C & Swift) & Android。我们可以在这些平台上共享Realm数据库文件,并且上层逻辑可以不用任何改动的情况下实现移植。

  • 高级
    Ream支持加密,格式化查询,易于移植,支持JSON,流式api,数据变更通知等高级特性

  • 可视化
    Realm 还提供了一个轻量级的数据库查看工具,在Mac Appstore 可以下载“Realm Browser”这个工具,开发者可以查看数据库当中的内容,执行简单的插入和删除数据的操作。

使用

1. 添加依赖

  • projectbuild 中加入依赖
buildscript {
  repositories {
      jcenter()
  }
  dependencies {
      classpath "io.realm:realm-gradle-plugin:2.2.1"
  }
}

在这里插入图片描述

  • module 中加入
apply plugin: 'realm-android'

2. 创建 model

创建一个 User 类,需要继承 RealmObject 。支持public, protected和 private的类以及方法

public class User extends RealmObject {
    private String name;
    private int age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}

除了直接继承于 RealmObject 来声明 Realm 数据模型之外,还可以通过实现 RealmModel 接口并添加 @RealmClass 修饰符来声明。

@RealmClass
public class User implements RealmModel {
    ...
}

3. 初始化

使用默认配置:

		Realm.init(this);
        Realm mRealm = Realm.getDefaultInstance();

这时候会创建一个叫做 default.realm的Realm文件,一般来说,这个文件位于/data/data/包名/files/。通过realm.getPath()来获得该Realm的绝对路径。

注意:模拟器上运行时,Realm.getDefaultInstance()抛出异常,真机上没问题

当然,我们还可以使用 RealmConfiguration 来配置 Realm

RealmConfiguration config = new RealmConfiguration.Builder() 
            .name("myrealm.realm") //文件名
            .schemaVersion(0) //版本号
            .build();
Realm realm = Realm.getInstance(config);

4. 关闭 Realm

记得使用完后,在 onDestroy 中关闭 Realm

@Override 
protected void onDestroy() { 
    super.onDestroy();
    // Close the Realm instance. 
    realm.close(); 
}

5. 版本升级

当数据结构发生变化是,需要升级数据库。对于Realm来说,数据库升级就是迁移操作,把原来的数据库迁移到新结构的数据库

例1:User类发生变化,移除age,新增个@Required的id字段。
User版本:version 0

String name;
int    age;

User版本:version 1

@Required
String    id;
String name;

创建迁移类 CustomMigration,需要实现RealmMigration接口。执行版本升级时的处理:

/**
  * 升级数据库
  */
 class CustomMigration implements RealmMigration {
     @Override
     public void migrate(DynamicRealm realm, long oldVersion, long newVersion) {
         RealmSchema schema = realm.getSchema();
         if (oldVersion == 0 && newVersion == 1) {
             RealmObjectSchema personSchema = schema.get("User");
             //新增@Required的id
             personSchema
                     .addField("id", String.class, FieldAttribute.REQUIRED)
                     .transform(new RealmObjectSchema.Function() {
                         @Override
                         public void apply(DynamicReal
mObject obj) {
                             obj.set("id", "1");//为id设置值
                         }
                     })
                     .removeField("age");//移除age属性
             oldVersion++;
         }
     }
 }

使用Builder.migration升级数据库,将版本号改为1(原版本号:0)。当Realm发现新旧版本号不一致时,会自动使用该迁移类完成迁移操作。

RealmConfiguration config = new RealmConfiguration.Builder() 
         .name("myrealm.realm") //文件名
         .schemaVersion(1) 
         .migration(new CustomMigration())//升级数据库
         .build();

6. 增

写入操作需要在事务中进行,可以使用executeTransaction方法来开启事务。

  • 使用executeTransaction方法插入数据
mRealm.executeTransaction(new Realm.Transaction() {
         @Override
         public void execute(Realm realm) {
             User user = realm.createObject(User.class);
             user.setName("Gavin");
             user.setAge(23);
         }
     });

注意:如果在UI线程中插入过多的数据,可能会导致主线程拥塞。

  • 使用copyToRealmOrUpdate或copyToRealm方法插入数据
    当Model中存在主键的时候,推荐使用copyToRealmOrUpdate方法插入数据。如果对象存在,就更新该对象;反之,它会创建一个新的对象。若该Model没有主键,使用copyToRealm方法,否则将抛出异常。
final User user = new User();
      user.setName("Jack");
      user.setId("2");
      mRealm.executeTransaction(new Realm.Transaction() {
          @Override
          public void execute(Realm realm) {
              realm.copyToRealmOrUpdate(user);
          }
      });
  • 使用executeTransactionAsync该方法会开启一个子线程来执行事务,并且在执行完成后进行结果通知。
RealmAsyncTask transaction = mRealm.executeTransactionAsync(new Realm.Transaction() {
 @Override
 public void execute(Realm realm) {
     User user = realm.createObject(User.class);
     user.setName("Eric");
     user.setId("4");
   }
});

注意:如果当Acitivity或Fragment被销毁时,在OnSuccess或OnError中执行UI操作,将导致程序奔溃 。用RealmAsyncTask .cancel();可以取消事务

7. 删

  • 使用deleteFromRealm()
//先查找到数据
final RealmResults userList = mRealm.where(User.class).findAll();
mRealm.executeTransaction(new Realm.Transaction() {
  @Override
  public void execute(Realm realm) {
      userList.get(0).deleteFromRealm();
  }
});
  • 使用deleteFromRealm(int index)
mRealm.executeTransaction(new Realm.Transaction() {
  @Override
  public void execute(Realm realm) {
      userList.deleteFromRealm(0);
  }
});

8. 改

mRealm.executeTransaction(new Realm.Transaction() {
    @Override
    public void execute(Realm realm) {
        //先查找后得到User对象
        User user = mRealm.where(User.class).findFirst();
        user.setAge(26);
    }
});

9. 查

  • findAll ——查询
RealmResults userList = mRealm.where(User.class).findAll();
  • findAllAsync——异步查询
RealmResults userList = mRealm.where(User.class)
              .equalTo("name", "Kevin")
              .findAllAsync();
  • findFirst ——查询第一条数据
User user2 = mRealm.where(User.class).findFirst();
  • equalTo ——根据条件查询
RealmResults userList = mRealm.where(User.class)
         .equalTo("name", "Kevin").findAll();
  • equalTo ——多条件查询
RealmResults userList = mRealm.where(User.class)
         .equalTo("name", "Kevin").findAll();
RealmResults userList = user5.where()
         .equalTo("dogs.name", "二哈").findAll();

想了解更多请查看 :
encryptionExample


http://www.niftyadmin.cn/n/272622.html

相关文章

边听歌边充电LDR6028+LDR9201既能充电又能OTG方案

随着type-c接口的普及,市面上的手机,平板,笔电逐渐都采用了type-c接口,设备为了不断的追求更轻薄的机身和防水要求慢慢的取消了一些影响手机外观完整性的接口,比如3.5mm耳机孔。 有线耳机用户一般会选择使用C口转3.5m…

二十三种设计模式第二篇--工厂模式

上篇我们了解了6条设计模式的准则,我相信如果你想了解设计模式,那么你迈出的第一步,我会将上一篇文档里边的6大准则进行一篇有关的代码展示,当然这是题外话了,本篇我们将重点围绕工厂模式进行讲解,天哪&…

OpenGL(一)——初识和搭建

目录 一、前言 二、概述 2.1 光学 2.2 三通道 2.3 上下文Context 2.4 渲染管线 2.5 着色器Shader 2.6 缓冲区和数组 三、安装 四、运行 五、库API 5.1 核心库GL 5.2 实用库GLUT 一、前言 渲染render是用软件从模型生成图像的过程,也表示编辑视频生成想达…

c# 数据保存为PDF(一) (spire pdf篇)

文章目录 前言了解 Spire使用Spire.PDF1 创建简单的PDF文档2 创建带有格式的PDF文档(使用Draw)头部信息页眉页脚测试数据完整的代码 3 创建带有格式的PDF文档(使用Gird)小结 先上一个效果图 前言 项目中需要将一些数据转存为PDF …

毕业小白的进阶之路 day1

摆烂的大学生活 学校里学到最多的也就是低阶算法以及python了,操作系统以及其他的就是学了忘压根不会应用。接触到最深的也就是Springboot了。学几年前淘汰的SSM框架真不知道是为了什么。最近临近毕业想要提升下自己,也就get到了Spring Cloud 框架。 S…

从功能测试转型测试开发,薪资涨了20K,1000字讲述转型必经之路...

身处职场之中,犹如逆水行舟不进则退,想要不被后浪拍死在沙滩上,就要不断学习新知识,接受新事物。 要得到更好的发展,就要紧跟发展趋势,不断转型才能保持竞争力,在职场中占有一席之地。 转型不…

bootloaders

什么是BootLoader? 一般来说,bootloader是一种软件/固件,它在SoC上电后立即运行。bootloader的主要职责是启动软件的后续部分,例如操作系统、baremetal应用程序或在某些情况下另一个bootloader。当涉及到嵌入式时,bootloader通常…

chatgpt模拟机器人软件开发

ChatGPT的参数取决于具体的模型和实现方式,但以下是一些常见的ChatGPT参数: 模型深度:指模型中神经网络的层数。通常情况下,层数越多,模型的表达能力也就越强。 隐藏单元大小:指在模型中每个隐藏层…