探秘鸿蒙 HarmonyOS NEXT:鸿蒙存储核心技术全解析

引言

本文章基于HarmonyOS NEXT操作系统,API12以上的版本。

ArkTS (ArkUI 框架) 中,用户首选项 (Preferences)持久化存储 (PersistentStorage) 都用于数据存储,但它们有不同的应用场景和特点。


1. 用户首选项 (Preferences)

概念

  • 轻量级键值对存储,主要用于存储用户设置或小型数据,如应用配置、主题模式、语言等。
  • 适用于 频繁读取少量更新 的数据存储。

特点

  • 主要用于存储简单的键值对,如 booleanstringnumber 等。
  • 数据存储在 本地沙盒 中,并会在 应用重新启动后保持不变
  • 适合存储 小量数据,如 UI 设置、开关状态等。

使用场景

  • 主题模式(浅色/深色模式)。
  • 语言偏好设置(中文/英文)。
  • 记住用户是否已登录(布尔值)。
  • 开关设置,如通知开关。

示例代码

import preferences from '@ohos.data.preferences';// 获取 Preferences 实例
async function saveThemePreference(isDarkMode: boolean) {const preferencesInstance = await preferences.getPreferences(globalThis, 'user_settings');await preferencesInstance.put('dark_mode', isDarkMode);await preferencesInstance.flush(); // 确保数据写入
}// 读取 Preferences
async function getThemePreference(): Promise<boolean> {const preferencesInstance = await preferences.getPreferences(globalThis, 'user_settings');return await preferencesInstance.get('dark_mode', false); // 默认值 false
}

2. 核心API介绍

Preferences 组件主要提供 读取、写入、删除 用户首选项的方法。常用的 API 主要包括:

(1)获取 Preferences 实例

在 OpenHarmony 设备上,用户首选项存储在 文件 中,因此需要指定存储路径。

import preferences from '@ohos.data.preferences';async function getPreferences() {try {const pref = await preferences.getPreferences(globalThis.getContext(), 'settings.preferences');return pref;} catch (err) {console.error('获取 Preferences 失败:', err);}
}
  • getPreferences(context, fileName): 通过 文件名 获取 Preferences 实例,该文件存储在应用的数据目录下。
  • context: 需要传入全局应用上下文 globalThis.getContext()

(2)存储数据

使用 put() 方法存储键值对数据,并使用 flush() 进行同步保存。

async function savePreferences() {const pref = await getPreferences();if (!pref) return;pref.put('theme', 'dark');  // 存储字符串pref.put('volume', 50);      // 存储数值pref.put('notifications', true);  // 存储布尔值await pref.flush();  // 立即保存到本地console.log('数据已保存');
}
  • put(key, value): 存储 key-value 键值对。
  • flush(): 同步 写入存储,否则数据可能不会立即写入。

(3)读取数据

使用 get() 方法获取存储的数据:

async function loadPreferences() {const pref = await getPreferences();if (!pref) return;const theme = pref.get('theme', 'light');  // 默认值 "light"const volume = pref.get('volume', 10);     // 默认值 10const notifications = pref.get('notifications', false);  // 默认值 falseconsole.log(`当前主题: ${theme}, 音量: ${volume}, 通知开关: ${notifications}`);
}
  • get(key, defaultValue): 读取 存储的值,如果键不存在,则返回默认值。

(4)删除数据

使用 delete() 方法移除指定键:

async function removePreferenceKey() {const pref = await getPreferences();if (!pref) return;pref.delete('theme');  // 删除 "theme" 配置项await pref.flush();  // 确保数据同步更新console.log('主题设置已删除');
}
  • delete(key): 删除指定 key 对应的值。

(5)清除所有数据

可以使用 clear() 方法清空整个 Preferences。

async function clearPreferences() {const pref = await getPreferences();if (!pref) return;pref.clear();  // 清空所有数据await pref.flush();console.log('所有用户设置已清除');
}
  • clear(): 清空所有存储的键值对数据。

3. 持久化存储 (PersistentStorage)

概念

  • 更通用的数据存储方式,可以用于存储更大规模的结构化数据,如对象、列表或复杂数据。
  • 通常与本地数据库(如 SQLite文件存储)结合使用。

特点

  • 适用于 存储较大数据量,如缓存数据、用户数据、应用状态等。
  • 提供更丰富的 API,支持复杂的数据管理。
  • 数据存储在应用的私有目录,可以长期存储,即使应用重启也不会丢失。

使用场景

  • 用户账户信息(如用户名、头像)。
  • 应用的缓存数据(如搜索历史、下载列表)。
  • 复杂的设置数据(如 JSON 结构的配置)。
  • 离线数据存储(如离线文档、任务列表)。

示例代码(基于数据存储组件)

import { PersistentStorage } from '@ohos.data.storage';// 保存数据
async function saveUserData(userData: any) {const storage = new PersistentStorage('user_data'); // 创建存储实例await storage.set('profile', JSON.stringify(userData));await storage.flush(); // 确保数据写入
}// 读取数据
async function getUserData(): Promise<any> {const storage = new PersistentStorage('user_data');const userDataStr = await storage.get('profile', '{}'); // 默认值为空对象return JSON.parse(userDataStr);
}

4. 核心API介绍

(1)存储数据

使用 set() 方法存储数据:

storage.set('username', 'Alice');
storage.set('age', 25);
storage.set('isPremiumUser', true);
  • 该方法支持存储 字符串、数值、布尔值等基本数据类型
  • 数据存储在应用本地,不会在应用重启时丢失。

(2)获取数据

使用 get<T>() 方法获取存储的数据:

let username: string = storage.get<string>('username', 'defaultUser');
let age: number = storage.get<number>('age', 0);
let isPremium: boolean = storage.get<boolean>('isPremiumUser', false);
  • 参数
    • 第一个参数是键(key)。
    • 第二个参数是 默认值(当 key 不存在时返回此值)。
  • 返回值类型 需要显式指定(泛型 <T>)。

(3)删除数据

使用 delete() 方法删除某个键值:

storage.delete('username');
  • 删除后,使用 get() 方法查询时将返回默认值。

(4)清空所有数据

使用 clear() 方法删除所有存储的键值对:

storage.clear();
  • 该方法会清除 PersistentStorage 中存储的所有数据,谨慎使用!

(5)数据是否存在

使用 has() 方法检查某个键是否存在:

if (storage.has('username')) {console.log('用户名已存储');
} else {console.log('用户名不存在');
}

(6)简单封装

以下是一个简单封装的 PersistentStorage 工具类:

import { PersistentStorage } from '@ohos.data.preferences';class StorageUtil {private static instance: PersistentStorage;private constructor() {}/*** 获取 PersistentStorage 实例(单例模式)*/private static getInstance(): PersistentStorage {if (!this.instance) {this.instance = PersistentStorage.getSharedInstance();}return this.instance;}/*** 存储数据* @param key 键名* @param value 要存储的值(支持 string、number、boolean)*/static set<T>(key: string, value: T): void {this.getInstance().set(key, value);}/*** 获取存储数据* @param key 键名* @param defaultValue 默认值(如果 key 不存在,则返回此值)* @returns 存储的数据*/static get<T>(key: string, defaultValue: T): T {return this.getInstance().get<T>(key, defaultValue);}/*** 删除指定 key 的存储数据* @param key 键名*/static delete(key: string): void {this.getInstance().delete(key);}/*** 清除所有存储数据*/static clear(): void {this.getInstance().clear();}/*** 判断某个 key 是否存在* @param key 键名* @returns 存在返回 true,否则返回 false*/static has(key: string): boolean {return this.getInstance().has(key);}
}export default StorageUtil;// 示例使用
StorageUtil.set('username', 'Alice');
const username = StorageUtil.get('username', 'defaultUser');
console.log(`Username: ${username}`);
StorageUtil.delete('username');
console.log(StorageUtil.has('username')); // false
StorageUtil.clear();

5. 对比总结

特性用户首选项 (Preferences)持久化存储 (PersistentStorage)
存储方式轻量级键值对存储适用于存储复杂数据
数据类型string, boolean, number复杂数据结构(JSON、列表等)
数据量小量数据大量数据
适用场景主题、语言、用户偏好用户信息、缓存、离线数据
读写性能读取速度快适用于批量存储
数据持久性持久化存储持久化存储

6. 使用建议

  • 如果只是存储 简单的设置数据,比如开关状态、主题模式,建议使用 Preferences
  • 如果需要存储 复杂对象(如 JSON 数据)或者 大量数据(如历史记录、缓存),建议使用 PersistentStorage

当然如果你的数据规模较大,还可以考虑 数据库 (键值型数据库KVManager或者关系型数据库RdbStore) 进行存储,以提高查询和管理效率。

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

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

相关文章

Leetcode—15. 三数之和(哈希表—基础算法)

题目&#xff1a; 给你一个整数数组 nums &#xff0c;判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i ! j、i ! k 且 j ! k &#xff0c;同时还满足 nums[i] nums[j] nums[k] 0 。请你返回所有和为 0 且不重复的三元组。 注意&#xff1a;答案中不可以包含重复的…

Linux 启动Jar脚本设置开机自启【超级详细】

Linux 启动Jar脚本&&设置开机自启【超级详细】 概要服务器开机自启服务重启脚本 概要 最近在Linux服务器中部署了一个项目&#xff08;单机版&#xff09;&#xff0c;每次更新服务的时候需要用到好几个命令&#xff0c;停止服务&#xff0c;再重启&#xff0c;并且服…

【第21节】windows sdk编程:网络编程基础

目录 引言&#xff1a;网络编程基础 一、socket介绍(套接字) 1.1 Berkeley Socket套接字 1.2 WinSocket套接字 1.3 WSAtartup函数 1.4 socket函数 1.5 字节序转换 1.6 绑定套接字 1.7 监听 1.8 连接 1.9 接收数据 1.10 发送数据 1.11 关闭套接字 二、UDP连接流程…

QT 图表(拆线图,栏状图,饼状图 ,动态图表)

效果 折线图 // 创建折线数据系列// 创建折线系列QLineSeries *series new QLineSeries;// series->append(0, 6);// series->append(2, 4);// series->append(3, 8);// 创建图表并添加系列QChart *chart new QChart;chart->addSeries(series);chart->setTit…

vector和list的区别是什么

vector 和 list 都是C 标准模板库&#xff08;STL&#xff09;中的容器&#xff0c;它们的区别如下&#xff1a; 内存结构 - vector &#xff1a;是连续的内存空间&#xff0c;就像数组一样&#xff0c;元素在内存中依次排列。 - list &#xff1a;是由节点组成的双向链表…

【AI】【AIGC】降低AIGC检测率:技术、挑战与应对策略

引言 随着生成式人工智能&#xff08;AIGC&#xff09;技术的迅速发展&#xff0c;越来越多的内容开始由人工智能生成。AIGC技术的应用非常广泛&#xff0c;包括文本生成、图像生成、音频生成等。然而&#xff0c;随着这些技术的普及&#xff0c;如何有效识别并检测AIGC生成的…

vue3 ts 请求封装后端接口

一 首页-广告区域-小程序 首页-广告区域-小程序 GET/home/banner1.1 请求封装 首页-广告区域 home.ts export const getHomeBannerApi (distributionSite 1) > {return http<BannerItem[]>({method: GET,url: /home/banner,data: {distributionSite,},}) }函数定…

响应式CMS架构优化SEO与用户体验

内容概要 在数字化内容生态中&#xff0c;响应式CMS架构已成为平衡搜索引擎可见性与终端用户体验的核心载体。该系统通过多终端适配技术&#xff0c;确保PC、移动端及平板等设备的内容渲染一致性&#xff0c;直接降低页面跳出率并延长用户停留时长。与此同时&#xff0c;智能S…

算法基础篇(1)(蓝桥杯常考点)

算法基础篇 前言 算法内容还有搜索&#xff0c;数据结构&#xff08;进阶&#xff09;&#xff0c;动态规划和图论 数学那个的话大家也知道比较难&#xff0c;放在最后讲 这期包含的内容可以看目录 模拟那个算法的话就是题说什么写什么&#xff0c;就不再分入目录中了 注意事…

MyBatis一级缓存和二级缓存

介绍 在开发基于 MyBatis 的应用时&#xff0c;缓存是提升性能的关键因素之一。MyBatis 提供了一级缓存和二级缓存&#xff0c;合理使用它们可以显著减少数据库的访问次数&#xff0c;提高系统的响应速度和吞吐量。本文将深入探讨 MyBatis 一级缓存和二级缓存的工作原理、使用…

C++核心语法快速整理

前言 欢迎来到我的博客 个人主页:北岭敲键盘的荒漠猫-CSDN博客 本文主要为学过多门语言玩家快速入门C 没有基础的就放弃吧。 全部都是精华&#xff0c;看完能直接上手改别人的项目。 输出内容 std::代表了这里的cout使用的标准库&#xff0c;避免不同库中的相同命名导致混乱 …

如何让自动驾驶汽车“看清”世界?坐标映射与数据融合概述

在自动驾驶领域,多传感器融合技术是实现车辆环境感知和决策控制的关键。其中,坐标系映射和对应是多传感器融合的重要环节,它涉及到不同传感器数据在统一坐标系下的转换和匹配,以实现对车辆周围环境的准确感知。本文将介绍多传感器融合中坐标系映射和对应的数学基础和实际应…

Unity Shader 的编程流程和结构

Unity Shader 的编程流程和结构 Unity Shader 的编程主要由以下三个核心部分组成&#xff1a;Properties&#xff08;属性&#xff09;、SubShader&#xff08;子着色器&#xff09; 和 Fallback&#xff08;回退&#xff09;。下面是它们的具体作用和结构&#xff1a; 1. Pr…

第十四天- 排序

一、排序的基本概念 排序是计算机科学中一项重要的操作&#xff0c;它将一组数据元素按照特定的顺序&#xff08;如升序或降序&#xff09;重新排列。排序算法的性能通常通过时间复杂度和空间复杂度来衡量。在 Python 中&#xff0c;有内置的排序函数&#xff0c;同时也可以手…

移除idea External Liraries 中maven依赖包

问题背景 扩展包里面不停的出现已经在POM文件注释的包&#xff0c;其实是没有查询到根源位置。 在IDEA插件中搜索Maven Helper 点击pom.xml文件 会出现扩展插件 定位之后在pom中添加exclusions&#xff0c;如下代码 <dependency><groupId>com.disney.eva.framewo…

AI革命!蓝耘携手海螺AI视频,打造智能化视频新纪元

AI革命&#xff01;蓝耘携手海螺AI视频&#xff0c;打造智能化视频新纪元 前言 在这个信息爆炸的时代&#xff0c;视频已经成为我们获取信息、学习新知识的重要方式。而随着人工智能&#xff08;AI&#xff09;技术的快速发展&#xff0c;AI与视频内容的结合为我们带来了全新的…

dify1.1.1安装

1、 按照GitHub上操作 下载源码&#xff0c;没有安装git的&#xff0c;可以下载成zip包&#xff0c; unzip 解压 git clone https://github.com/langgenius/dify.git cd dify cd docker cp .env.example .env2、启动前 &#xff0c;先改下 docker-compose.yaml&#xff0c;…

ElasticSearch 可观测性最佳实践

ElasticSearch 概述 ElasticSearch 是一个开源的高扩展的分布式全文检索引擎&#xff0c;它可以近乎实时的存储、检索数据&#xff1b;本身扩展性很好&#xff0c;可以扩展到上百台服务器&#xff0c;处理 PB 级别&#xff08;大数据时代&#xff09;的数据。ES 也使用 Java 开…

Excel处理控件Spire.XLS系列教程:C# 在 Excel 中添加或删除单元格边框

单元格边框是指在单元格或单元格区域周围添加的线条。它们可用于不同的目的&#xff0c;如分隔工作表中的部分、吸引读者注意重要的单元格或使工作表看起来更美观。本文将介绍如何使用 Spire.XLS for .NET 在 C# 中添加或删除 Excel 单元格边框。 安装 Spire.XLS for .NET E-…

前端Wind CSS面试题及参考答案

目录 标准盒模型与 IE 盒模型的区别是什么?如何通过 box-sizing 属性切换这两种盒模型? 如何计算一个元素在标准盒模型下的总宽度(包含 margin、padding、border)? 父元素高度塌陷的原因是什么?请列举至少 3 种清除浮动的方法。 方法一:使用 clear 属性 方法二:使用…