0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

鸿蒙的数据库知识点学习

OpenHarmony技术社区 来源:鸿蒙技术社区 作者:中软国际 2021-09-06 09:34 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

移动端开发,数据存储是非常重要的,鸿蒙也不例外,说到数据存储,首要的就是数据库了,数据库的存储机制是否完善,提供的功能是否简单方便,直接影响开发者的开发速度和性能。

作为鸿蒙开发者,最近就深入学习了鸿蒙的数据库知识点,了解了存储机制并且尝试了使用,发现鸿蒙的数据库真的做到了应有尽有,操作还简单的地步。

鸿蒙关系型数据库

先来看看鸿蒙的关系型数据库(Relational Database,RDB)概念。

鸿蒙的 RDB 是一种基于关系模型来管理数据的数据库。HarmonyOS 关系型数据库基于 SQLite 组件提供了一套完整的对本地数据库进行管理的机制,对外提供了一系列的增、删、改、查接口,也可以直接运行用户输入的 SQL 语句来满足复杂的场景需要。

HarmonyOS 提供的关系型数据库功能更加完善,查询效率更高。

概念中一句话很重要,HarmonyOS 提供的关系型数据库功能更加完善,查询效率更高。

看到这里我当时是非常激动的,作为开发者难道不是最希望使用的 API 在什么情况下都适用吗?

下面看看鸿蒙数据库的运作机制,了解机制才能了解数据库开发的核心,也有利于扩展。

运作机制

HarmonyOS 关系型数据库对外提供通用的操作接口,底层使用 SQLite 作为持久化存储引擎,支持 SQLite 具有的所有数据库特性,包括但不限于事务、索引、视图、触发器、外键、参数化查询和预编译 SQL 语句。

看到上面的鸿蒙数据库的运行机制不难发现,主要工作还是在 framework 层做的封装,然后调用 JNI,使用的还是 SQLite 组件。

不管底层怎么做的开发,只要功能完善,体验到位,鸿蒙应用端直接使用不是很香吗?

约束与限制

①数据库中连接池的最大数量是 4 个,用以管理用户的读写操作。

连接池的数量是有限制的 ,最大时 4 个,不过 4 个已经足够使用了。

②为保证数据的准确性,数据库同一时间只能支持一个写操作。

同一时间支持一个写操作时非常重要的,为了防止数据存储的正确性,鸿蒙做了这一个限制,但是作为多年的移动端开发者,一般这种多操作或者大数据操作,都会使用多线程,异步线程,或者放在线程池中,这样就更完美了。

数据库操作 DataAbility

鸿蒙在创建类的时候有一个 DataAbility,不知道各位开发者使用过了没,其实这个就是为了数据库操作而来的。

添加步骤很简单:添加类的时候选择 Empty DataAbility 即可。

①配置

添加类后,会自动生成如下配置:

{

“permissions”: [

“com.huawei.codelab.DataAbilityShellProvider.PROVIDER”

],

“name”: “com.hadiidbouk.databasemanager.database.DataAbility”,

“icon”: “$media:icon”,

“description”: “$string:dataability_description”,

“type”: “data”,

“uri”: “dataability://com.huawei.codelab.PersonDataAbility”

}

操作数据库需要权限信息:

“permissions”: [

“com.huawei.codelab.DataAbilityShellProvider.PROVIDER”

需要配置 url,url 很重要,在进行数据库表操作的时候需要保持一致。

②DataAbility 操作内容

默认创建的 DataAbility 类会自动重写 数据库的增,删,改,查 几种操作的函数。

可以看下面:

添加 DataAbility 会自动重写四个接口函数,有关数据库的增,删,改,查。

该 DataAbility 在运行项目后会自行执行 onStart 方法进行数据库及其数据表的创建工作。

通过 RdbPredicates 数据库进行数据库关系的关联进行操作。

public class DataAbility extends Ability {

private static final HiLogLabel LABEL_LOG = new HiLogLabel(3, 0xD001100, “Demo”);

private static final String DB_NAME = “persondataability.db”;

private static final String DB_TAB_NAME = “person”;

private static final String DB_COLUMN_PERSON_ID = “id”;

private static final String DB_COLUMN_NAME = “name”;

private static final String DB_COLUMN_GENDER = “gender”;

private static final String DB_COLUMN_AGE = “age”;

private static final int DB_VERSION = 1;

private StoreConfig config = StoreConfig.newDefaultConfig(DB_NAME);

private RdbStore rdbStore;

private RdbOpenCallback rdbOpenCallback = new RdbOpenCallback() {

@Override

public void onCreate(RdbStore store) {

store.executeSql(“create table if not exists ”

+ DB_TAB_NAME + “ (”

+ DB_COLUMN_PERSON_ID + “ integer primary key, ”

+ DB_COLUMN_NAME + “ text not null, ”

+ DB_COLUMN_GENDER + “ text not null, ”

+ DB_COLUMN_AGE + “ integer)”);

}

@Override

public void onUpgrade(RdbStore store, int oldVersion, int newVersion) {

}

};

@Override

public void onStart(Intent intent) { //创建数据库操作

super.onStart(intent);

HiLog.info(LABEL_LOG, “DataAbility onStart”);

DatabaseHelper databaseHelper = new DatabaseHelper(this);

rdbStore = databaseHelper.getRdbStore(config, DB_VERSION, rdbOpenCallback, null);

}

// 数据库 查询操作

@Override

public ResultSet query(Uri uri, String[] columns, DataAbilityPredicates predicates) {

RdbPredicates rdbPredicates = DataAbilityUtils.createRdbPredicates(predicates, DB_TAB_NAME);

ResultSet resultSet = rdbStore.query(rdbPredicates, columns);

if (resultSet == null) {

HiLog.info(LABEL_LOG, “resultSet is null”);

}

return resultSet;

}

// 数据库 插入操作

@Override

public int insert(Uri uri, ValuesBucket value) {

HiLog.info(LABEL_LOG, “DataAbility insert”);

String path = uri.getLastPath();

if (!“person”.equals(path)) {

HiLog.info(LABEL_LOG, “DataAbility insert path is not matched”);

return -1;

}

ValuesBucket values = new ValuesBucket();

values.putInteger(DB_COLUMN_PERSON_ID, value.getInteger(DB_COLUMN_PERSON_ID));

values.putString(DB_COLUMN_NAME, value.getString(DB_COLUMN_NAME));

values.putString(DB_COLUMN_GENDER, value.getString(DB_COLUMN_GENDER));

values.putInteger(DB_COLUMN_AGE, value.getInteger(DB_COLUMN_AGE));

int index = (int) rdbStore.insert(DB_TAB_NAME, values);

DataAbilityHelper.creator(this, uri).notifyChange(uri);

return index;

}

// 数据库 删除操作

@Override

public int delete(Uri uri, DataAbilityPredicates predicates) {

RdbPredicates rdbPredicates = DataAbilityUtils.createRdbPredicates(predicates, DB_TAB_NAME);

int index = rdbStore.delete(rdbPredicates);

HiLog.info(LABEL_LOG, “delete: ” + index);

DataAbilityHelper.creator(this, uri).notifyChange(uri);

return index;

}

// 数据库 更新操作

@Override

public int update(Uri uri, ValuesBucket value, DataAbilityPredicates predicates) {

RdbPredicates rdbPredicates = DataAbilityUtils.createRdbPredicates(predicates, DB_TAB_NAME);

int index = rdbStore.update(value, rdbPredicates);

HiLog.info(LABEL_LOG, “update: ” + index);

DataAbilityHelper.creator(this, uri).notifyChange(uri);

return index;

}

@Override

public FileDescriptor openFile(Uri uri, String mode) {

return null;

}

@Override

public String[] getFileTypes(Uri uri, String mimeTypeFilter) {

return new String[0];

}

@Override

public PacMap call(String method, String arg, PacMap extras) {

return null;

}

@Override

public String getType(Uri uri) {

return null;

}

}

③数据库操作

这里的数据库操作时,开发需求做的数据库操作,可以通过自己的需求来开发数据库的调用操作,最终还是通过使用 DataAbility 直接调用系统的数据库。

public class DataBaseAbilitySlice extends AbilitySlice {

private static final HiLogLabel LABEL_LOG = new HiLogLabel(3, 0xD001100, “Demo”);

private DataAbilityHelper databaseHelper;

private static final String BASE_URI = “dataability:///com.huawei.codelab.PersonDataAbility”;

private static final String DATA_PATH = “/person”;

private static final String DB_COLUMN_PERSON_ID = “id”;

private static final String DB_COLUMN_NAME = “name”;

private static final String DB_COLUMN_GENDER = “gender”;

private static final String DB_COLUMN_AGE = “age”;

@Override

public void onStart(Intent intent) {

super.onStart(intent);

super.setUIContent(ResourceTable.Layout_ability_data_base);

databaseHelper = DataAbilityHelper.creator(this);

Text text = (Text)findComponentById(ResourceTable.Id_text_helloworld);

text.setClickedListener(new Component.ClickedListener() {

@Override

public void onClick(Component component) {

query();

insert(100, “Tom”, “male”, 20);

insert(101, “Jerry”, “female”, 21);

insert(102, “Bob”, “male”, 22);

query(); // 查看插入后的结果

update();

query(); // 查看更新后的结果

delete();

query(); // 查看删除后的结果

}

});

}

@Override

public void onActive() {

super.onActive();

}

@Override

public void onForeground(Intent intent) {

super.onForeground(intent);

}

private void insert(int id, String name, String gender, int age) {

ValuesBucket valuesBucket = new ValuesBucket();

valuesBucket.putInteger(DB_COLUMN_PERSON_ID, id);

valuesBucket.putString(DB_COLUMN_NAME, name);

valuesBucket.putString(DB_COLUMN_GENDER, gender);

valuesBucket.putInteger(DB_COLUMN_AGE, age);

try {

if (databaseHelper.insert(Uri.parse(BASE_URI + DATA_PATH), valuesBucket) != -1) {

// if (databaseHelper.insert(Uri.parse(BASE_URI + DATA_PATH), valuesBucket) != -1) {

HiLog.info(LABEL_LOG, “insert successful”);

}

} catch (DataAbilityRemoteException | IllegalStateException exception) {

HiLog.error(LABEL_LOG, “insert: dataRemote exception|illegalStateException”);

}

}

private void delete() {

DataAbilityPredicates predicates = new DataAbilityPredicates()

.equalTo(DB_COLUMN_PERSON_ID, 100);

try {

if (databaseHelper.delete(Uri.parse(BASE_URI + DATA_PATH), predicates) != -1) {

HiLog.info(LABEL_LOG, “delete successful”);

}

} catch (DataAbilityRemoteException | IllegalStateException exception) {

HiLog.error(LABEL_LOG, “delete: dataRemote exception | illegalStateException”);

}

}

private void update() {

DataAbilityPredicates predicates = new DataAbilityPredicates();

predicates.equalTo(DB_COLUMN_PERSON_ID, 102);

ValuesBucket valuesBucket = new ValuesBucket();

valuesBucket.putString(DB_COLUMN_NAME, “ZhangSanPlus”);

valuesBucket.putInteger(DB_COLUMN_AGE, 28);

try {

if (databaseHelper.update(Uri.parse(BASE_URI + DATA_PATH), valuesBucket, predicates) != -1) {

HiLog.info(LABEL_LOG, “update successful”);

}

} catch (DataAbilityRemoteException | IllegalStateException exception) {

HiLog.error(LABEL_LOG, “update: dataRemote exception | illegalStateException”);

}

}

private void query() {

String[] columns = new String[] {DB_COLUMN_PERSON_ID,

DB_COLUMN_NAME, DB_COLUMN_GENDER, DB_COLUMN_AGE};

// 构造查询条件

DataAbilityPredicates predicates = new DataAbilityPredicates();

predicates.between(DB_COLUMN_AGE, 15, 40); // 查询时间段

try {

ResultSet resultSet = databaseHelper.query(Uri.parse(BASE_URI + DATA_PATH),

columns, predicates);

if (resultSet == null || resultSet.getRowCount() == 0) {

HiLog.info(LABEL_LOG, “query: resultSet is null or no result found”);

return;

}

resultSet.goToFirstRow();

do {

int id = resultSet.getInt(resultSet.getColumnIndexForName(DB_COLUMN_PERSON_ID));

String name = resultSet.getString(resultSet.getColumnIndexForName(DB_COLUMN_NAME));

String gender = resultSet.getString(resultSet.getColumnIndexForName(DB_COLUMN_GENDER));

int age = resultSet.getInt(resultSet.getColumnIndexForName(DB_COLUMN_AGE));

HiLog.info(LABEL_LOG, “query: Id :” + id + “ Name :” + name + “ Gender :” + gender + “ Age :” + age);

} while (resultSet.goToNextRow());

} catch (DataAbilityRemoteException | IllegalStateException exception) {

HiLog.error(LABEL_LOG, “query: dataRemote exception | illegalStateException”);

}

}

}

数据插入使用对象类 ValuesBucket;使用 DataAbilityPredicates 实例进行数据查询的条件设置。

到此有关数据库的关系型数据库操作基本就完成了,是不是非常,非常简单,可以直接拿来主义验证一下。

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 数据库
    +关注

    关注

    7

    文章

    4083

    浏览量

    68544
  • 中软国际
    +关注

    关注

    0

    文章

    751

    浏览量

    8212
  • 鸿蒙系统
    +关注

    关注

    183

    文章

    2642

    浏览量

    70124
  • HarmonyOS
    +关注

    关注

    80

    文章

    2157

    浏览量

    36294

原文标题:鸿蒙的关系型数据库,被惊艳到了!

文章出处:【微信号:gh_834c4b3d87fe,微信公众号:OpenHarmony技术社区】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    开发知识库测试添加知识库

    文档类型的知识要等待数据校验完成后才能上架 可以点击知识名称查看知识详情 等待后端处理完成可以点击知识列表的上架 在智能体中
    发表于 03-06 15:07

    鸿蒙智能体开发知识库---创建知识库

    在小艺智能体平台页面,通过【工作空间】-【知识库】-【新建知识库】,进入新建知识库流程。 若勾选【授权知识库用于知识问答,授权后该
    发表于 03-06 10:18

    Oracle数据库ASM实例无法挂载的数据恢复案例

    一个Oracle数据库故障表现为ASM磁盘组掉线,ASM实例无法挂载(mount)。数据库管理员自行进行简单修复,未能成功,随后联系北亚数据恢复中心恢复数据
    的头像 发表于 02-24 15:19 207次阅读
    Oracle<b class='flag-5'>数据库</b>ASM实例无法挂载的<b class='flag-5'>数据</b>恢复案例

    深度解读MySQL数据库备份恢复策略

    数据是企业的核心资产,数据库备份是保障数据安全的最后一道防线。某天上午10,运营人员误执行了一条DELETE语句,删除了订单表中近一个月的数据
    的头像 发表于 01-26 17:40 855次阅读

    国产数据库的AI战事

    国产数据库硝烟再起,Vastbase V100构筑企业智能基座
    的头像 发表于 10-24 20:45 4416次阅读
    国产<b class='flag-5'>数据库</b>的AI战事

    数据库性能优化指南

    作为一名在大厂摸爬滚打多年的运维老兵,我见过太多因为数据库性能问题导致的生产事故。今天分享一套完整的数据库优化方法论,从SQL层面到硬件配置,帮你彻底解决性能瓶颈!
    的头像 发表于 08-18 11:21 903次阅读

    数据库数据恢复—服务器异常断电导致Oracle数据库故障的数据恢复案例

    Oracle数据库故障: 某公司一台服务器上部署Oracle数据库。服务器意外断电导致数据库报错,报错内容为“system01.dbf需要更多的恢复来保持一致性”。该Oracle数据库
    的头像 发表于 07-24 11:12 845次阅读
    <b class='flag-5'>数据库</b><b class='flag-5'>数据</b>恢复—服务器异常断电导致Oracle<b class='flag-5'>数据库</b>故障的<b class='flag-5'>数据</b>恢复案例

    三款主流国产数据库的技术特点

    随着数字经济的快速发展和数据安全要求的提升,国产数据库正迎来前所未有的发展机遇。在信创浪潮推动下,达梦数据库、TiDB、华为高斯数据库等国产数据库
    的头像 发表于 07-14 11:08 1345次阅读

    数据库数据恢复—MongoDB数据库文件丢失的数据恢复案例

    MongoDB数据库数据恢复环境: 一台操作系统为Windows Server的虚拟机上部署MongoDB数据库。 MongoDB数据库故障: 工作人员在MongoDB服务仍
    的头像 发表于 07-01 11:13 785次阅读
    <b class='flag-5'>数据库</b><b class='flag-5'>数据</b>恢复—MongoDB<b class='flag-5'>数据库</b>文件丢失的<b class='flag-5'>数据</b>恢复案例

    数据库数据恢复—SQL Server数据库被加密如何恢复数据

    SQL Server数据库故障: SQL Server数据库被加密,无法使用。 数据库MDF、LDF、log日志文件名字被篡改。
    的头像 发表于 06-25 13:54 843次阅读
    <b class='flag-5'>数据库</b><b class='flag-5'>数据</b>恢复—SQL Server<b class='flag-5'>数据库</b>被加密如何恢复<b class='flag-5'>数据</b>?

    达梦数据库常用管理SQL命令详解

    达梦数据库常用管理SQL命令详解
    的头像 发表于 06-17 15:12 7654次阅读
    达梦<b class='flag-5'>数据库</b>常用管理SQL命令详解

    oracle数据恢复—oracle数据库误执行错误truncate命令如何恢复数据

    oracle数据库误执行truncate命令导致数据丢失是一种常见情况。通常情况下,oracle数据库误操作删除数据只需要通过备份恢复数据
    的头像 发表于 06-05 16:01 1793次阅读
    oracle<b class='flag-5'>数据</b>恢复—oracle<b class='flag-5'>数据库</b>误执行错误truncate命令如何恢复<b class='flag-5'>数据</b>?

    SQLSERVER数据库是什么

    SQL Server 是由微软公司开发的一款 关系型数据库管理系统(RDBMS) ,用于存储、管理和检索结构化数据。它是企业级应用中广泛使用的数据库解决方案之一,尤其适用于Windows平台,但也
    的头像 发表于 05-26 09:19 1303次阅读

    MySQL数据库是什么

    MySQL数据库是一种 开源的关系型数据库管理系统(RDBMS) ,由瑞典MySQL AB公司开发,后被Oracle公司收购。它通过结构化查询语言(SQL)进行数据存储、管理和操作,广泛应用于Web
    的头像 发表于 05-23 09:18 1450次阅读

    电机选型计算公式与知识点汇总

    纯分享帖,需要者可点击附件获取完整资料~~~*附件:电机选型计算公式与知识点汇总.pdf 【免责声明】内容转自今日电机,因转载众多,无法确认真正原始作者,故仅标明转载来源。版权归原出处所有,纯分享帖,侵权请联系删除内容以保证您的权益。
    发表于 04-29 16:10