鸿蒙开发实战-如何开发一个字符串加解密应用程序

news/2024/4/28 19:21:00/文章来源:https://blog.csdn.net/2401_82546228/article/details/137089095

介绍

本Codelab针对用户隐私安全,使用加密算法API对密码进行加密存储,模拟开发一个用户注册登录应用。实现如下功能:

  1. 实现登录、注册、登录成功页面。
  2. 注册的用户数据保存到关系型数据库。
  3. 登录时通过查询数据库校验用户是否存在、密码是否正确。
  4. 密码通过加密算法保存和使用。

相关概念

  • 加解密算法库框架:为屏蔽底层硬件和算法库,向上提供统一的密码算法库加解密相关接口。
  • 关系型数据库:关系型数据库(Relational Database,RDB)是一种基于关系模型来管理数据的数据库。

环境搭建

软件要求

  • DevEco Studio版本:DevEco Studio 3.1 Release。
  • OpenHarmony SDK版本:API version 9。

硬件要求

  • 开发板类型:润和RK3568开发板。
  • OpenHarmony系统:3.2 Release。

环境搭建

完成本篇Codelab我们首先要完成开发环境的搭建,本示例以RK3568开发板为例,参照以下步骤进行:

  1. 获取OpenHarmony系统版本:标准系统解决方案(二进制)。以3.2 Release版本为例:

2.搭建烧录环境。

  • 完成DevEco Device Tool的安装
  • 完成RK3568开发板的烧录

3.搭建开发环境。

  • 开始前请参考工具准备,完成DevEco Studio的安装和开发环境配置。
  • 开发环境配置完成后,请参考使用工程向导创建工程(模板选择“Empty Ability”)。
  • 工程创建完成后,选择使用真机进行调测。

代码结构解读

本篇Codelab只对核心代码进行讲解。

├──entry/src/main/ets                   // 代码区
│  ├──common
│  │  ├──constants
│  │  │  └──CommonConstants.ets         // 公共常量类
│  │  └──utils
│  │     ├──AesUtil.ets                 // 加解密工具类
│  │     ├──DataTransformUtil.ets       // 数据转换工具类
│  │     ├──Logger.ets                  // 日志打印工具类
│  │     └──PromptUtil.ts               // 弹窗工具类
│  ├──entryability
│  │  └──EntryAbility.ts                // 程序入口类
│  ├──model
│  │  ├──RdbModel.ets                   // 数据库业务处理文件
│  │  └──UserTableApi.ets               // 用户表具体业务文件
│  ├──pages
│  │  ├──Login.ets                      // 登录页
│  │  ├──Register.ets                   // 注册页
│  │  └──Welcome.ets                    // 欢迎页
│  └──viewmodel
│     └──User.ets                       // 用户实体类
└──entry/src/main/resources             // 资源文件目录

关系型数据库

首先编写创建表的SQL语句,其中user为表名、id为主键并自动递增、username为用户名、password为加密后的密码、authTag为加解密认证信息。

// CommonConstants.ets/*** 创建表的SQL语句*/static readonly CREATE_TABLE_SQL: string = 'CREATE TABLE IF NOT EXISTS user(' +'id INTEGER PRIMARY KEY AUTOINCREMENT, ' +'username TEXT NOT NULL, ' +'password TEXT NOT NULL, ' +'authTag TEXT NOT NULL)';

在RdbModel的构造方法中,调用getRdbStore方法创建数据库。其中STORE_CONFIG为数据库相关配置,sqlCreateTable为创建user用户表所需的SQL语句。

// RdbModel.ets
import dataRdb from '@ohos.data.relationalStore';
...
export class RdbModel {private rdbStore: dataRdb.RdbStore | null = null;private tableName: string = '';private sqlCreateTable: string = '';...constructor(tableName: string, sqlCreateTable: string, columns: Array<string>) {this.tableName = tableName;this.sqlCreateTable = sqlCreateTable;this.columns = columns;this.getRdbStore();}/*** 获取数据库操作对象rdbStore.*/getRdbStore() {let getPromiseRdb = dataRdb.getRdbStore(getContext(), { name: CommonConstants.DATABASE_NAME, securityLevel: dataRdb.SecurityLevel.S1 });getPromiseRdb.then(rdbStore => {this.rdbStore = rdbStore;this.rdbStore.executeSql(this.sqlCreateTable);}).catch((err: Error) => {Logger.error(`getRdbStore err ${JSON.stringify(err)}`);});}

创建UserTableApi.ets文件,实例化RdbModel创建userTable对象。并对外提供可操作用户数据表的API接口,包括插入数据、根据用户名查询用户信息等方法。

// UserTableApi.ets
export class UserTableApi {private userTable = new RdbModel(TABLE_NAME, CREATE_USER_TABLE, COLUMNS);/*** 将数据保存到数据库中** @param user 需要保存的User类型的数据对象*/insertUserData(user: User) {this.userTable.insertData(user);}/*** 根据用户名查询用户信息** @param username 查询的用户名* @returns 查询结果集*/async queryUserByUsername(username: string): Promise<User[]> {let resultList: Array<User>;// 过滤条件let predicates = new dataRdb.RdbPredicates(TABLE_NAME);predicates.equalTo('username', username);// 将查询到的结果封装成User对应的用户信息let ret = await this.userTable.query(predicates);resultList = this.getListFromResultSet(ret);return resultList;}...
}

密码加解密

创建AesUtil工具类,封装加解密相关逻辑。首先引入@ohos.security.cryptoFramework包,在构造方法中初始化加解密算法框架所需的环境,包括密钥规格的选择、加解密规格的选择等。本示例采用对称AES加解密算法,密钥长度为256位,分组模式为GCM。具体有以下步骤:

  1. 创建对称密钥生成器。
  2. 通过密钥生成器生成对称密钥。
  3. 创建加解密生成器。
  4. 通过加解密生成器加密或解密数据。

说明: 对于对称密钥、非对称密钥、加解密算法模式,具体可见**密钥生成规格**。

// AesUtil.ets
import cryptoFramework from '@ohos.security.cryptoFramework';
...
class AesUtil {private globalCipher: cryptoFramework.Cipher = cryptoFramework.createCipher(CommonConstants.GENERATOR_NAME;private globalKey: cryptoFramework.SymKey = null;/*** 构造函数初始化加解密环境、生成密钥*/constructor() {let symAlgName = CommonConstants.ENCRYPTION_MODE;// 创建对称密钥生成器let symKeyGenerator = cryptoFramework.createSymKeyGenerator(symAlgName);// 通过密钥生成器和keyMaterialBlob生成256位长度的对称密钥let keyMaterialBlob = this.genKeyMaterialBlob(CommonConstants.KEY_DATA);symKeyGenerator.convertKey(keyMaterialBlob, (err, symKey) => {if (err) {Logger.error(`Convert symKey failed, ${err.code}, ${err.message}`);return;}this.globalKey = symKey;let cipherAlgName = CommonConstants.GENERATOR_NAME;try {// 生成加解密生成器this.globalCipher = cryptoFramework.createCipher(cipherAlgName);} catch (error) {Logger.error(`createCipher failed, error is ${JSON.stringify(err)}`);}});}// 加密async encrypt(content: string, authTag: string): Promise<string> {...}// 解密 async decrypt(content: string, authTag: string): Promise<string> {...}
}

密码加密

由于加密算法采用GCM分组模式,在加密前需要获取GCM模式加解密所需的参数GcmParamsSpec。依次生成长度为12字节、8字节、16字节的DataBlob类型的数据,并封装成GcmParamsSpec对象。

// AesUtil.ets
class AesUtil {.../*** 获取GCM分组加解密所需的参数** @returns 返回加密所需参数的promise实例*/async genGcmParamsSpec(): Promise<cryptoFramework.GcmParamsSpec> {let ivBlob: cryptoFramework.DataBlob = this.genKeyMaterialBlob(CommonConstants.GCM_IV_DATA);let aadBlob: cryptoFramework.DataBlob = this.genKeyMaterialBlob(CommonConstants.GCM_AAD_DATA);let tagBlob: cryptoFramework.DataBlob = this.genKeyMaterialBlob(CommonConstants.GCM_TAG_DATA);let gcmParamsSpec: cryptoFramework.GcmParamsSpec = {iv: ivBlob,aad: aadBlob,authTag: tagBlob,algName: `GcmParamsSpec`};return gcmParamsSpec;}/*** 根据数据组生成DataBlob类型的数据** @param data 需要封装的数据* @returns Blob DataBlob类型的数据*/genKeyMaterialBlob(data: Array<number>): cryptoFramework.DataBlob {let keyMaterial = new Uint8Array(data);return { data: keyMaterial };}
}

在AesUtil.ets的encrypt方法中实现密码加密逻辑。由于本示例加密数据量较小,所以这里直接使用update一步完成加密操作。若数据量较大,可通过update方法分段加密。主要实现以下步骤:

  1. 调用Cipher的init方法初始化加密环境。设置mode为ENCRYPT_MODE,传入密钥和生成的gcmParams。
  2. 将用户输入的密码转换为Uint8Array数组,进而封装成DataBlob对象。
  3. 将封装好的plainTextBlob传入update方法中完成加密。
  4. 调用doFinal方法,传入null,取出加密后的认证信息authTag。
  5. 取出Uint8Array类型的加密数据和authTag,转换成Base64类型的字符串,封装成User类型返回。
// AesUtil.ets
class AesUtil {.../*** 加密** @param content 加密内容* @returns 返回携带密文User对象的promise实例*/async encrypt(content: string): Promise<User> {// 初始化加密环境let mode = cryptoFramework.CryptoMode.ENCRYPT_MODE;let gcmParams = await this.genGcmParamsSpec();await this.globalCipher.init(mode, this.globalKey, gcmParams);let plainTextBlob = {// 字符串转Uint8Arraydata: DataTransformUtil.stringToUint8Array(content)};// 加密let updateOutput: cryptoFramework.DataBlob = await this.globalCipher.update(plainTextBlob);if (!updateOutput) {return Promise.reject('encrypt updateOutput is null');}let authTag: cryptoFramework.DataBlob = await this.globalCipher.doFinal(null);// Uint8Array转base64let encryptContent: string = DataTransformUtil.uint8ArrayToBase64(updateOutput.data);let authTagContent: string = DataTransformUtil.uint8ArrayToBase64(authTag.data);let user = new User(null, ``, encryptContent, authTagContent);return user;}
}

密码解密

解密操作与加密类似,主要实现以下步骤:

  1. 调用Cipher的init方法初始化解密环境,设置mode为DECRYPT_MODE,传入携带authTag认证信息的gcmParams。
  2. 将Base64类型密文转换为Uint8Array数组,进而封装成DataBlob对象。
  3. 将封装好的plainTextBlob传入doFinal方法中完成解密。
  4. 取出Uint8Array类型的解密数据,转换成字符串并返回。
// AesUtil.ets
class AesUtil {.../*** 解密** @param content 解密内容* @param authTag GCM 解密所需认证信息内容* @returns 返回解密内容的promise实例*/async decrypt(content: string, authTag: string): Promise<string> {// 初始化解密环境let mode = cryptoFramework.CryptoMode.DECRYPT_MODE;let gcmParams = await this.genGcmParamsSpec();let authTagBlob: cryptoFramework.DataBlob = {data: DataTransformUtil.base64ToUint8Array(authTag)};gcmParams.authTag = authTagBlob;await this.globalCipher.init(mode, this.globalKey, gcmParams);let plainTextBlob: cryptoFramework.DataBlob = {// base64转Uint8Arraydata: DataTransformUtil.base64ToUint8Array(content)};// 解密let finalOutput: cryptoFramework.DataBlob = await this.globalCipher.doFinal(plainTextBlob);if (!finalOutput) {return Promise.reject('decrypt finalOutput is null');}// Uint8Array转字符串let decryptContent = DataTransformUtil.uint8ArrayToString(finalOutput.data);return decryptContent;}
}

总结

您已经完成了本次Codelab的学习,并了解到以下知识点:

  1. 加解密算法框架的使用。
  2. 关系型数据库的使用。

为了帮助大家更深入有效的学习到鸿蒙开发知识点,小编特意给大家准备了一份全套最新版的HarmonyOS NEXT学习资源,获取完整版方式请点击→HarmonyOS教学视频

HarmonyOS教学视频:语法ArkTS、TypeScript、ArkUI等…视频教程

鸿蒙生态应用开发白皮书V2.0PDF:

获取完整版白皮书方式请点击→《鸿蒙生态应用开发白皮书V2.0PDF

在这里插入图片描述

鸿蒙 (Harmony OS)开发学习手册

一、入门必看

  1. 应用开发导读(ArkTS)
  2. .……

在这里插入图片描述


二、HarmonyOS 概念

  1. 系统定义
  2. 技术架构
  3. 技术特性
  4. 系统安全

在这里插入图片描述

三、如何快速入门?《鸿蒙基础入门学习指南》

  1. 基本概念
  2. 构建第一个ArkTS应用
  3. .……

在这里插入图片描述


四、开发基础知识

  1. 应用基础知识
  2. 配置文件
  3. 应用数据管理
  4. 应用安全管理
  5. 应用隐私保护
  6. 三方应用调用管控机制
  7. 资源分类与访问
  8. 学习ArkTS语言
  9. .……

在这里插入图片描述


五、基于ArkTS 开发

  1. Ability开发
  2. UI开发
  3. 公共事件与通知
  4. 窗口管理
  5. 媒体
  6. 安全
  7. 7.网络与链接
  8. 电话服务
  9. 数据管理
  10. 后台任务(Background Task)管理
  11. 设备管理
  12. 设备使用信息统计
  13. DFX
  14. 国际化开发
  15. 折叠屏系列
  16. .……

在这里插入图片描述


更多了解更多鸿蒙开发的相关知识可以参考:《鸿蒙 (Harmony OS)开发学习手册

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

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

相关文章

如何高效系统地自学 Python?

导言&#xff1a; Python作为一门流行的编程语言&#xff0c;被广泛运用于数据分析、人工智能、网络应用等领域。想要系统地自学Python&#xff0c;并掌握其核心概念和编程技能&#xff0c;需要一定的方法和步骤。本文将介绍如何高效系统地自学Python&#xff0c;让你能够快速…

docker推拉时的数据交换详解

前言 docker用了这么久了, 有没有想过, 在执行docker push 和 docker pull命令的时候, 数据是如何传递的呢? 换句话说, 如果要实现一个镜像仓库, 针对推拉的服务, 如何实现接口呢? 根据OCI 分发规范文档 的描述, 已经对整个推拉过程中要调用的接口有描述了. 但是, 纸上学来…

Linux升级GCC

文章目录 一、安装 EPEL 仓库二、更新yum三、安装 CentOS 开发工具组四、安装scl五、安装gcc 11六、启用gcc 11七、设置永久使用 一、安装 EPEL 仓库 命令&#xff1a; yum install epel-release -y二、更新yum 命令&#xff1a; yum update -y三、安装 CentOS 开发工具组 …

蓝桥杯练习题总结(三)线性dp题(摆花、数字三角形加强版)

目录 一、摆花 思路一&#xff1a; 确定状态&#xff1a; 初始化&#xff1a; 思路二&#xff1a; 确定状态&#xff1a; 初始化&#xff1a; 循环遍历&#xff1a; 状态转移方程&#xff1a; 二、数字三角形加强版 一、摆花 题目描述 小明的花店新开张&#xff0c;为了吸…

python知识点总结(十)

python知识点总结十 1、装饰器的理解、并实现一个计时器记录执行性能&#xff0c;并且将执行结果写入日志文件中2、队列和栈的区别&#xff0c;并且用python实现3、设计实现遍历目录与子目录4、CPU处理进程最慢的情况通常发生在以下几种情况下&#xff1a;5、CPU处理线程最慢的…

删除数组中的指定元素(了解如何删除数组中的指定元素,并返回一个新的数组,看这一篇就足够了!)

前言&#xff1a;有时候我们会遇到要在数组中删除指定元素&#xff0c;但是不能创建新的数组&#xff0c;那么这个时候应该如何操作呢&#xff1f; ✨✨✨这里是秋刀鱼不做梦的BLOG ✨✨✨想要了解更多内容可以访问我的主页秋刀鱼不做梦-CSDN博客 废话不多讲&#xff0c;让我们…

脚本实现Ubuntu设置屏幕无人操作,自动黑屏

使用 xrandr 命令可以实现对屏幕的控制&#xff0c;包括调整分辨率、旋转屏幕以及关闭屏幕等。要实现 Ubuntu 设置屏幕在无人操作一段时间后自动黑屏&#xff0c;非待机&#xff0c;并黑屏后点击触摸屏可以唤醒屏幕&#xff0c;可以借助 xrandr 命令来实现。 首先&#xff0c;…

基于ssm在线云音乐系统的设计与实现论文

摘 要 随着移动互联网时代的发展&#xff0c;网络的使用越来越普及&#xff0c;用户在获取和存储信息方面也会有激动人心的时刻。音乐也将慢慢融入人们的生活中。影响和改变我们的生活。随着当今各种流行音乐的流行&#xff0c;人们在日常生活中经常会用到的就是在线云音乐系统…

macos配置maven

Mac Maven 安装及配置 - 知乎 官网上下载一个zip 配置环境变量vim ~/.bash_profile 我打开来看到之前配过conda的&#xff0c;和教程里不一样。那就在之前的配置下方添加就好了。 既然你的.bash_profile文件中已经有了一些配置&#xff0c;特别是Anaconda的初始化脚本&#…

鸿蒙HarmonyOS应用开发之Rawfile开发指导

场景介绍 开发者可以通过本指导了解在OpenHarmony应用中&#xff0c;如何使用Native Rawfile接口操作Rawfile目录和文件。功能包括文件列表遍历、文件打开、搜索、读取和关闭Rawfile。 接口说明 接口名描述NativeResourceManager *OH_ResourceManager_InitNativeResourceMan…

哪些属于“法律、行政法规另有规定,依照其规定进行评估/批准”的情况?

哪些属于“法律、行政法规另有规定&#xff0c;依照其规定进行评估/批准”的情况&#xff1f; 除《网络安全法》《数据安全法》和《个人信息保护法》确立的数据和网络安全整体体系外&#xff0c;企业还应当考虑其他相关法律法规的要求。 例如&#xff1a; ✮如根据《中华人民…

OpenHarmony实战开发-滑动容器组件Swiper的使用

介绍 本篇Codelab主要介绍了滑动容器组件Swiper的几种常见的应用场景&#xff0c;包括顶部导航、轮播图以及视频滑动播放。 相关概念 Swiper&#xff1a;滑动容器&#xff0c;提供子组件切换滑动的能力。Stack&#xff1a;堆叠容器&#xff0c;子组件按照顺序依次入栈&#x…

训练svm并部署树莓派

训练svm并部署树莓派 开发环境1. 准备数据集2. 训练模型3. 部署模型开发环境 vscode python 3.8 用到的库: scikit-learn==1.3.2 pickle torch pandas matplotlib 1. 准备数据集 数据为xls文件,如下格式 2. 训练模型 文件结构 执行训练 python代码 import pickle &…

【计算机网络】IP 协议

网络层IP协议 一、认识 IP 地址二、IP 协议报头格式三、网段划分1. 初识子网划分2. 理解子网划分3. 子网掩码4. 特殊的 IP 地址5. IP 地址的数量限制6. 私有 IP 地址和公网 IP 地址7. 理解全球网络&#xff08;1&#xff09;理解公网&#xff08;2&#xff09;理解私网&#xf…

Git 常用命令速查

Git 是一个分布式版本控制系统&#xff0c;用于管理代码和其他文件。它允许您跟踪代码的更改&#xff0c;并在必要时回滚到以前的版本。 本文将介绍一些 Git 常用命令&#xff0c;帮助您快速上手 Git。 初始化 Git 仓库 git init添加文件到暂存区 git add <file_name>…

【正版特惠】IDM 永久授权 优惠低至109元!

尽管小编有修改版IDM&#xff0c;但是由于软件太好用了&#xff0c;很多同学干脆就直接购买了正版&#xff0c;现在正版也不贵&#xff0c;并且授权码绑定自己的邮箱&#xff0c;直接官方下载激活&#xff0c;无需其他的绿化修改之类的操作&#xff0c;不喜欢那么麻烦的&#x…

简易指南:国内ip切换手机软件怎么弄

在网络访问受到地域限制的情况下&#xff0c;使用国内IP切换手机软件可以帮助用户轻松访问被屏蔽的内容&#xff0c;扩展网络体验。以下是虎观代理小二分享的使用国内IP切换手机软件的简易指南。并提供一些注意事项。 如何在手机上使用国内IP切换软件 步骤一&#xff1a;选择I…

16.JRE和JDK

程序员在编写代码的时候其实是需要一些环境&#xff0c;例如我们之前写的HelloWorld。我们需要的东西有JVM、核心类库、开发工具。 1、JVM&#xff08;Java Virtual Machine&#xff09;&#xff1a;Java虚拟机&#xff0c;真正运行Java程序的地方。没有虚拟机&#xff0c;代码…

R使用netmeta程序包实现对罕见事件(Rare events)的网状meta分析

在进行网状meta分析过程中&#xff0c;一些试验经常会出现罕见事件&#xff08;Rare event&#xff09;。尤其是在安全性评价中&#xff0c;由于一些不良事件发生率低、样本量不充足&#xff0c;导致试验组和对照组的事件发生例数少&#xff0c;甚至出现0事件。针对出现0事件的…

【任职资格】某大型制造型企业任职资格体系项目纪实

该企业以业绩、责任、能力为导向&#xff0c;确定了分层分类的整体薪酬模式&#xff0c;但是每一名员工到底应该拿多少工资&#xff0c;同一个岗位的人员是否应该拿同样的工资是管理人员比较头疼的事情。华恒智信顾问认为&#xff0c;通过任职资格评价能实现真正的人岗匹配&…