鸿蒙应用开发中的数据存储:SQLite与Preferences全面解析

在鸿蒙(HarmonyOS)应用开发中,数据存储是构建功能完整、用户体验良好的应用程序的关键环节。鸿蒙系统提供了多种数据存储解决方案,其中SQLite数据库和Preferences(偏好设置)是最常用的两种方式。本文将深入探讨这两种存储技术的使用场景、实现方法以及最佳实践,帮助开发者根据应用需求选择最合适的存储方案。

一、数据存储概述

在移动应用开发中,数据存储通常分为以下几种类型:

  1. 轻量级键值存储:如Preferences,适合存储简单的配置信息

  2. 结构化数据存储:如SQLite,适合存储复杂的关系型数据

  3. 文件存储:适合存储大容量非结构化数据

  4. 云存储:用于需要跨设备同步的数据

鸿蒙系统为开发者提供了全面的数据存储API,其中Preferences和SQLite因其易用性和高效性成为最常用的本地存储方案。

二、Preferences(偏好设置)详解

2.1 Preferences简介

Preferences是鸿蒙提供的一种轻量级数据存储解决方案,它以键值对的形式存储数据,适用于保存用户偏好设置、应用配置信息等小规模数据。

核心特点:

  • 采用XML文件格式存储

  • 线程安全

  • 支持基本数据类型(String、int、boolean等)

  • 数据量不宜过大(建议不超过1MB)

2.2 Preferences实现方法

2.2.1 初始化Preferences

import ohos.data.preferences.Preferences;// 获取Preferences实例
// 参数1:上下文对象
// 参数2:存储文件名(无需后缀)
Preferences preferences = PreferencesHelper.getPreferences(context, "user_settings");

2.2.2 数据存取操作

存储数据:

// 存储不同类型的数据
preferences.putString("username", "鸿蒙开发者");
preferences.putInt("login_count", 5);
preferences.putBoolean("is_first_launch", false);
preferences.putFloat("font_size", 16.0f);
preferences.putLong("last_login_time", System.currentTimeMillis());// 提交保存(同步写入磁盘)
preferences.flush();

读取数据:

// 读取数据(第二个参数为默认值)
String username = preferences.getString("username", "");
int loginCount = preferences.getInt("login_count", 0);
boolean isFirstLaunch = preferences.getBoolean("is_first_launch", true);
float fontSize = preferences.getFloat("font_size", 14.0f);
long lastLoginTime = preferences.getLong("last_login_time", 0L);

删除数据:

// 删除单个键值
preferences.delete("username");// 清空所有数据
preferences.clear();

2.3 Preferences最佳实践

  1. 合理组织键名:使用有意义的命名,如"user_pref_theme_color"而非简单的"color"

  2. 避免存储大对象:不适合存储图片、大文本等数据

  3. 及时flush:重要数据操作后立即调用flush()确保数据持久化

  4. 分组管理:不同模块的配置使用不同的Preferences文件

  5. 加密敏感数据:对于密码等敏感信息应先加密再存储

三、SQLite数据库详解

3.1 SQLite简介

SQLite是鸿蒙内置的轻量级关系型数据库,具有以下特点:

  • 无需服务器,零配置

  • 支持标准SQL语法

  • 支持事务处理

  • 单文件存储

  • 适合存储结构化数据

3.2 SQLite实现方法

3.2.1 创建数据库帮助类

import ohos.data.rdb.*;public class UserDBHelper extends RdbOpenHelper {private static final String DB_NAME = "user_db";private static final int DB_VERSION = 2;// 用户表SQLprivate static final String CREATE_TABLE_USER = "CREATE TABLE IF NOT EXISTS user (" +"id INTEGER PRIMARY KEY AUTOINCREMENT, " +"name TEXT NOT NULL, " +"age INTEGER DEFAULT 0, " +"email TEXT UNIQUE, " +"create_time TEXT DEFAULT (datetime('now','localtime')))";public UserDBHelper(Context context) {super(context, DB_NAME, null, DB_VERSION);}@Overridepublic void onCreate(RdbStore store) {// 创建初始表结构store.executeSql(CREATE_TABLE_USER);}@Overridepublic void onUpgrade(RdbStore store, int oldVersion, int newVersion) {// 数据库升级逻辑if (oldVersion < 2) {// 版本2新增address字段store.executeSql("ALTER TABLE user ADD COLUMN address TEXT");}}
}

3.2.2 初始化数据库

// 数据库配置
StoreConfig config = StoreConfig.newDefaultConfig("user_db.db");
// 设置是否加密
config.setEncryptKey("your_encryption_key".getBytes());// 初始化数据库
UserDBHelper helper = new UserDBHelper(context);
RdbStore rdbStore = helper.getRdbStore(config);

3.2.3 CRUD操作实现

1. 插入数据

ValuesBucket values = new ValuesBucket();
values.putString("name", "李四");
values.putInteger("age", 28);
values.putString("email", "lisi@example.com");
values.putString("address", "北京市海淀区");long newId = rdbStore.insert("user", values);

2. 查询数据

// 构建查询条件
RdbPredicates predicates = new RdbPredicates("user").greaterThan("age", 20).orderByAsc("name").limit(10).offset(0);// 指定返回列
String[] columns = {"id", "name", "age", "email", "address"};// 执行查询
ResultSet resultSet = rdbStore.query(predicates, columns);// 处理结果集
List<User> userList = new ArrayList<>();
while (resultSet.goToNextRow()) {User user = new User();user.setId(resultSet.getInt(0));user.setName(resultSet.getString(1));user.setAge(resultSet.getInt(2));user.setEmail(resultSet.getString(3));user.setAddress(resultSet.getString(4));userList.add(user);
}
resultSet.close();

3. 更新数据

ValuesBucket updateValues = new ValuesBucket();
updateValues.putInteger("age", 29);RdbPredicates updatePredicates = new RdbPredicates("user").equalTo("email", "lisi@example.com");int updatedRows = rdbStore.update(updateValues, updatePredicates);

4. 删除数据

RdbPredicates deletePredicates = new RdbPredicates("user").lessThanOrEqualTo("age", 18);int deletedRows = rdbStore.delete(deletePredicates);

3.2.4 事务处理

rdbStore.beginTransaction();
try {// 批量插入数据for (User user : userList) {ValuesBucket values = new ValuesBucket();values.putString("name", user.getName());// 设置其他字段...rdbStore.insert("user", values);}// 标记事务成功rdbStore.markAsCommit();
} catch (Exception e) {// 发生异常回滚事务rdbStore.markAsRollback();throw e;
} finally {// 结束事务rdbStore.endTransaction();
}

3.3 SQLite最佳实践

  1. 合理设计表结构:遵循数据库规范化原则

  2. 使用索引优化查询:对频繁查询的字段创建索引

  3. 避免在主线程操作数据库:使用线程池或异步任务

  4. 处理数据库升级:妥善管理onUpgrade逻辑

  5. 使用预编译语句:提高频繁操作的性能

  6. 定期维护数据库:如VACUUM操作减少碎片

四、SQLite与Preferences对比选型

特性PreferencesSQLite
存储类型键值对关系型表格
适合场景简单配置/用户偏好复杂结构化数据
查询能力简单键值查找强大SQL查询
数据量小(KB级)大(GB级)
性能极高
线程安全需要自行管理
加密支持有限支持完整加密
学习曲线简单中等

选型建议:

  • 选择Preferences:用户主题设置、应用开关状态、简单的计数器等

  • 选择SQLite:用户通讯录、聊天记录、商品目录等复杂数据

五、安全注意事项

  1. 敏感数据加密:无论是Preferences还是SQLite,存储密码等敏感信息前必须加密

  2. 数据库加密:使用setEncryptKey()方法加密SQLite数据库

  3. 文件权限控制:确保数据文件不被其他应用访问

  4. 输入验证:防止SQL注入攻击

  5. 定期备份:重要数据应实现备份机制

六、总结

鸿蒙系统提供的Preferences和SQLite两种存储方案各有优势,开发者应根据实际需求选择合适的方案。Preferences操作简单、性能极高,适合存储少量简单数据;SQLite功能强大、查询灵活,适合管理复杂结构化数据。

在实际开发中,我们常常会同时使用这两种存储方式:用Preferences保存应用配置和用户偏好,用SQLite管理应用的核心业务数据。掌握这两种存储技术,能够为鸿蒙应用开发提供坚实的数据持久化支持。

随着鸿蒙生态的不断发展,数据存储技术也将持续演进。建议开发者定期关注鸿蒙官方文档,了解最新的存储API和改进特性,以构建更高效、更安全的鸿蒙应用程序。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/diannao/89048.shtml
繁体地址,请注明出处:http://hk.pswp.cn/diannao/89048.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

夏至之日,共赴实时 AI 之约:RTE Open Day@AGI Playground 2025 回顾

每年 RTE 开发者社区的重磅活动—— RTE Open Day &#xff0c;也在六月的 AGI Playground 现场开启今年的行程。这是 RTE Open Day 第五期现场&#xff0c;这期我们的关键词是 「Real-Time AI」 和 「Voice Agent」&#xff0c;不仅有来自社区的 16 个项目&#xff0c;还有两场…

Tomcat性能调优指南

文章目录 一、Tomcat性能调优概述为什么需要调优Tomcat&#xff1f; 二、Tomcat架构与性能关键点三、JVM调优1. 内存配置优化2. 垃圾回收优化3. 其他JVM优化参数 四、连接器(Connector)调优1. NIO vs APR/Native2. 高级NIO配置 五、线程池优化六、会话管理优化1. 会话超时配置2…

Swift 小技巧:用单边区间优雅处理模糊范围

进入正题之前先科普一下 Swift 区间的知识。 Swift 中的区间有两种类型&#xff1a;闭区间和半开区间。 闭区间&#xff1a;用 a...b 表示&#xff0c;包含 a 和 b。半开区间&#xff1a;用 a..<b 表示&#xff0c;包含 a 但不包含 b。 举个例子 想判断一个数字是否在 0 …

Tang Prime 20K板OV2640例程

准备用Tang Prime 20K开发板进行OV2640摄像头采集验证。 Tang Primer 20K是由开源硬件厂商SiPEED矽速科技推出&#xff0c;是一款以 GW2A-LV18PG256C8/I7 为主芯片的核心板&#xff0c;准备了 2 个扩展板&#xff0c;Dock 和 Lite。板卡包含有HDMI输出&#xff0c;DVP接口&…

基于Anaconda环境开发IntelliJ IDEA实用JSON转Java实体插件

在软件开发中&#xff0c;将JSON数据转换为Java实体类是常见需求。借助Anaconda环境强大的包管理能力与IntelliJ IDEA的插件开发体系&#xff0c;我们可以打造一款高效实用的JSON转Java实体插件&#xff0c;显著提升开发效率。下面将从需求分析、技术选型、开发实现到优化部署&…

idea运行到远程机器 和 idea远程JVM调试

一、idea运行到远程机器 适用场景&#xff0c;本地连接不上远程机器的部分组件&#xff0c;如&#xff1a;redis、数据库。 缺点&#xff1a;每次修改程序&#xff0c;会复制所有的 依赖和class 启动比较慢。 工作原理&#xff1a;远程机器和本机器&#xff0c;都会启动一个端口…

微信小程序接入腾讯云短信验证码流程

以下是针对 AA公司微信小程序接入腾讯云短信验证码 的 全流程操作指南&#xff0c;包含资质申请、签名/模板配置、代码对接的完整解决方案&#xff1a; 一、资质申请&#xff08;必须通过审核才能发短信&#xff09; 1️⃣ 进入资质管理页 路径&#xff1a;腾讯云控制台 → 短…

阿里云OSS文件上传完整实现方案

一、前言 阿里云对象存储服务(OSS)是一种海量、安全、低成本、高可靠的云存储服务。本文将详细介绍如何在Spring Boot项目中集成阿里云OSS实现文件上传功能。 二、准备工作 1. 获取OSS配置信息 在开始前&#xff0c;您需要准备以下OSS配置信息&#xff1a; endpoint: OSS服…

【软考--软件设计师】10.2 关系型数据库

10 模式分解 分解 模式分解:将一个关系模式分解为多个子模式 模式分解就是模式规范化的工具&#xff0c;模式分解使用无损连接和保持函数依赖来衡量模式分解后是否导致原有模式中部分信息丢失。 无损连接 保持函数依赖 11、事务管理 事务的ACID性质: (1)原子性(Atomicit…

python训练day44 预训练模型

预训练模型发展史 预训练模型的训练策略 import torch import torch.nn as nn import torch.optim as optim from torchvision import datasets, transforms from torch.utils.data import DataLoader import matplotlib.pyplot as plt# 设置中文字体支持 plt.rcParams["…

[论文阅读]MISSRce

论文title: MISSRec: Pre-training and Transferring Multi-modal Interest-aware Sequence Representation for Recommendation

Redis学习笔记——黑马点评 附近商铺到UV统计 完结

前言&#xff1a; 今天完结了Redis的所有实战篇。 学习收获&#xff1a; GEO数据结构&#xff1a; GEO就是Geolocation的简写形式&#xff0c;代表地理坐标。Redis在3.2版本中加入对Geo的支持&#xff0c;存储、管理和操作地理空间数据的特殊数据结构&#xff0c;它能高效处…

【客户端排查】mac电脑怎么查看客户端的实时运行日志

先退出客户端&#xff1b;打开访达里的应用程序&#xff1b; 打开【显示包内容】&#xff1b; 找到MacOS 双击里面的终端程序&#xff1b; 双击后&#xff0c;客户端会自动启动&#xff0c;且可以在终端中查看客户端的实时日志啦~

HarmonyOS NEXT仓颉开发语言实战案例:健身App

各位好&#xff0c;今日分享一个健身app的首页&#xff1a; 这个页面看起比之前的案例要稍微复杂一些&#xff0c;主要在于顶部部分&#xff0c;有重叠的背景&#xff0c;还有偏移的部分。重叠布局可以使用Stack容器实现&#xff0c;超出容器范围的偏移可以使用负数间距来实现&…

TreeMap源码分析 红黑树

今天尝试刨一下TreeMap的祖坟。 底层结构对比 先来看一下与HashMap、LinkedHashMap和TreeMap的对比&#xff0c;同时就当是复习一下&#xff1a; HashMap使用数组存储数据&#xff0c;并使用单向链表结构存储hash冲突数据&#xff0c;同一个冲突桶中数据量大的时候&#xff…

华为云Flexus+DeepSeek征文|基于Dify构建拍照识题智能学习助手

华为云FlexusDeepSeek征文&#xff5c;基于Dify构建拍照识题智能学习助手 一、构建拍照识题智能学习助手前言二、构建拍照识题智能学习助手环境2.1 基于FlexusX实例的Dify平台2.2 基于MaaS的模型API商用服务 三、构建拍照识题智能学习助手实战3.1 配置Dify环境3.2 配置Dify工具…

题解:CF2120E Lanes of Cars

根据贪心&#xff0c;不难想到每次会把最长队伍末尾的那辆车移动到最短队伍的末尾。但由于 k k k 的存在&#xff0c;会导致一些冗余移动的存在。设需要挪动 C C C 辆车&#xff0c;则怒气值可以表示为 f ( C ) k C f(C) kC f(C)kC&#xff0c;其中 f ( C ) f(C) f(C) 是…

Excel基础:选择和移动

本文演示Excel中基础的选择和移动操作&#xff0c;并在最后提供了一张思维导图&#xff0c;方便记忆。 文章目录 一、选择1.1 基础选择1.1.1 选择单个单元格1.1.2 选择连续范围 1.2 行列选择1.2.1 选择整行整列1.2.2 选择多行多列 1.3 全选1.3.1 全选所有单元格1.3.2 智能选择…

Java面试宝典:基础四

80. int vs Integer 维度intInteger类型基本数据类型(8种之一)包装类默认值0null应用场景性能敏感场景(计算密集)Web表单、ORM框架(区分null和0)特殊能力无提供工具方法(如parseInt())和常量(如MAX_VALUE)示例:

RabbitMQ + JMeter 深度集成指南:中间件性能优化全流程解析!

在 2025 年的数字化浪潮中&#xff0c;中间件性能直接决定系统的稳定性和用户体验&#xff0c;而 RabbitMQ 作为消息队列的“老大哥”&#xff0c;在分布式系统中扮演着关键角色。然而&#xff0c;高并发场景下&#xff0c;消息堆积、延迟激增等问题可能让系统不堪重负&#xf…