鸿蒙开发中的并发与多线程
文章目录
- 前言
- 异步并发 (Promise和async/await)
- 多线程并发
- 并发能力选择
- 耗时任务并发执行场景
- 常见业务场景
- 常驻任务并发执行场景
- 常见业务场景
- 传统共享内存并发业务
- 长时任务并发执行场景
- 常见业务场景
- 并发任务管理
- 线程间通信
- 同语言线程间通信(ArkTS内)
- 线程间模块共享(单例模式)
- 实现方案介绍(方案二)
- 线程间不可变数据共享
- 实现方案介绍
- 生产者与消费者模式
前言
并发是指在同一时间内,存在多个任务同时执行的情况。对于多核设备,这些任务可能同时在不同CPU上并行执行。对于单核设备,多个并发任务不会在同一时刻并行执行,但是CPU会在某个任务休眠或进行I/O操作等状态下切换任务,调度执行其他任务,提升CPU的资源利用率。
异步并发 (Promise和async/await)
- Promise 的特点与用法:Promise 用于处理异步操作,有 pending、fulfilled、rejected 三种状态,通过构造函数传入 executor 函数创建 Promise 对象,executor 函数接收 resolve 和 reject 回调函数。可使用 then 和 catch 方法指定不同状态的回调函数。
- async/await 的优势:async/await 是 Promise 的语法糖,使异步代码更易读,async 函数返回 Promise 对象,内部可用 await 等待 Promise 解析,将异步操作以同步方式编写,还可结合 try/catch 捕获异常。
多线程并发
-
ArkTS并发模型:ArkTS采用内存隔离的线程模型,不同线程之间通过消息通信,线程内无锁化运行。与传统共享内存并发模型相比,
异步I/O不阻塞ArkTS线程,TaskPool及I/O线程池由系统管理,提升能效。 -

不支持在多线程中使用AppStorage。 -
TaskPool的使用:TaskPool可执行耗时任务,提供任务分发入口,支持将任务分发到不同优先级队列,自动管理工作线程并根据任务数量扩缩容。适用于相对独立的耗时任务、长时任务,但任务执行中有通信开销、不能阻塞过长、不能有上下文依赖等限制。
-
Worker的使用:Worker可执行常驻任务,需开发者主动创建或关闭并维护其生命周期,同时运行的Worker子线程数量有限。适用于长耗时且并发量不大的常驻任务场景。
-
实现特点对比:
- 内存模型与参数传递机制:TaskPool 和 Worker 都采用线程间隔离、内存不共享的方式,且参数传递都采用结构化克隆算法,支持 ArrayBuffer 转移和 SharedArrayBuffer 共享。,开发者自行管理数量及生命周期,不支持上述 TaskPool 的其他特性。
-
适用场景对比:
- TaskPool 适用场景:大多数场景推荐使用,性能优于 Worker,适用于需要设置优先级、频繁取消任务、大量或调度点较分散任务等。
- Worker 适用场景:适用于运行时间超过 3 分钟的任务、有关联的一系列同步任务等特定场景。
场景编号 | 场景分类 | 场景名称 | 简述 |
1 | 并发能力选择 | 耗时任务并发执行 | 相对独立的耗时任务需要放到单独的子线程中执行,推荐TaskPool |
2 | 常驻任务并发执行 | 常驻的耗时任务需要放到单独的子线程中执行,推荐Worker | |
3 | 共享内存并发业务 | 开发常见的共享内存并发业务,推荐使用TaskPool和Worker的API进行开发 | |
4 | 长时任务并发执行 | 长时间运行的任务,不独占线程执行,推荐TaskPool长时任务 | |
5 | 并发任务管理 | 多任务关联执行(串行顺序依赖) | 有严格执行顺序的任务,不希望并发执行 |
6 | 多任务关联执行(树状依赖) | 待执行的任务存在依赖关系,等待被依赖执行完再调度 | |
7 | 多任务同步等待结果(任务组) | 多个关联的任务需要等待全部结果返回后再进行后续操作 | |
8 | 多任务优先级调度 | 不同任务设置不同的优先级 | |
9 | 任务延时调度 | 任务不希望立即执行,希望延时一定时间后调度 | |
10 | 线程间通信 | 同语言线程间通信(ArkTS内) | 介绍ArkTS线程间的通信机制 |
11 | 跨语言多线程通信(C++与ArkTS) | 介绍C++与ArkTS线程间的通信机制 | |
12 | 线程间模块共享(单例模式) | 介绍进程内单例场景的实现方式 | |
13 | 线程间不可变数据共享 | 介绍不可变数据共享场景的实现方式 | |
14 | 生产者与消费者模式 | 介绍生产者与消费者模式场景的实现方式 |
并发能力选择
耗时任务并发执行场景
典型的耗时任务有CPU密集型任务、I/O密集型任务以及同步任务。=>任务池(TaskPool)
对于独立的耗时任务,不建议采用Worker来实现。
import { taskpool } from '@kit.ArkTS';@Concurrent
async function foo(a: number, b: number) {return a + b;
}taskpool.execute(foo, 1, 2).then((ret: Object) => { // 结果处理console.log('Return:' + ret);
})
常见业务场景
| 常见业务场景 | 具体业务描述 |
|---|---|
| 图片/视频编解码 | 将图片或视频进行编解码再展示 |
| 压缩/解压缩 | 对本地压缩包进行解压操作,或者本地文件的压缩操作 |
| JSON解析 | 对JSON字符串的序列化和反序列化操作 |
| 模型运算 | 对数据进行模型运算分析等 |
| 网络下载 | 密集网络请求下载资源、图片、文件等 |
| 数据库操作 | 将聊天记录、页面布局信息、音乐列表信息等保存到数据库,或者应用二次启动时,读取数据库展示相关信息 |
常驻任务并发执行场景
常驻不是指可以在后台保活运行的任务,而是相比于短时任务,时间更长的任务,可能与主线程生命周期一致。
对于一些长耗时(大于3min)且并发量不大的常驻任务场景,使用Worker在后台线程中运行这些耗时逻辑,避免阻塞主线程而导致出现丢帧卡顿等影响用户体验性的问题 。
常驻任务不推荐作为任务分发给TaskPool。
常见业务场景
| 常见业务场景 | 具体业务描述 |
|---|---|
| 游戏中台场景 | 启动子线程作为游戏业务的主逻辑线程,UI线程只负责渲染 |
| 产线硬件压测 | 需要阻塞调用硬件能力,做老化测试,阻塞式 |
传统共享内存并发业务
如果需要使用内存共享,当前可以通过Node-API到C++层进行共享,或者定义Sendable对象进行线程间数据共享。
长时任务并发执行场景
长时任务不同于阻塞任务,长周期运行,但是每次执行不会阻塞线程很久。因此不推荐将需要独占线程的任务封装成长时任务。任务池(TaskPool)
对于非常驻的长时任务,不建议采用Worker来实现。
长时任务指的是长时间不间断运行的独立任务,例如监听某个事件,发起执行后不会再接收发起方的输入,虽然也可以使用worker(推荐常驻后台任务才使用worker),但是更推荐使用TaskPool,TaskPool更方便,资源消耗更低。
常见业务场景
| 常见业务场景 | 具体业务描述 |
|---|---|
| 定期传感器数据采集 | 周期性采集一些传感器信息(例如位置信息、速度传感器等),应用运行阶段常驻运行。 |
| Socket端口信息监听 | 长时间监听Socket数据,不定时需要响应处理。 |
import { taskpool } from '@kit.ArkTS';@Concurrent
async function foo() {// 长监听等任务taskpool.Task.sendData();
}function executeTaskPool() {let longTask: taskpool.LongTask = new taskpool.LongTask(foo);longTask.onReceiveData((msg: Object) => {// 监听回调console.info(`onReceiveData, ${JSON.stringify(msg)}`);});taskpool.execute(longTask).then(() => {console.info('execute');});
}executeTaskPool();
并发任务管理
多任务关联执行(串行顺序依赖)
多任务关联执行(树状依赖)
多任务同步等待结果(任务组)
多任务优先级调度
任务延时调度
线程间通信
同语言线程间通信(ArkTS内)
| 跨线程交互场景 | 通信方式 | 通信优先级 |
|---|---|---|
| 宿主JS线程->TaskPool线程 | 参数传递后分发任务;过程中不支持正向通信 | 支持 |
| TaskPool线程->宿主JS线程 | 结果返回;sendData触发宿主线程异步回调,底层为uv_async_send实现 | 不支持 |
| 宿主JS线程->Worker线程 | 采用postMessage&onmessage异步通信 | 不支持 |
| Worker线程->宿主JS线程 | 异步方式:采用postMessage & onmessage异步通信 同步方式:支持Worker线程同步调用宿主线程注册的方法,并返回结果 | 不支持 |
| 任意JS线程<->任意JS线程 | 使用ohos.emitter实现双向异步通信 | 支持 |
线程间模块共享(单例模式)
实现方案介绍(方案二)
步骤一:采用ArkTS对象,定义Sendable类的单例,封装成共享模块(进程内共享),子线程进行初始化;
步骤二:初始化完成通知主线程,主线程导入使用该单例对象。
线程间不可变数据共享
实现方案介绍
通过冻结API,使共享对象变成只读对象。实现方案介绍:
步骤一:业务逻辑定义、生成需要的Sendable对象;
步骤二:发送到其他ArkTS线程前通过Object.Freeze API冻结该对象;
步骤三:通过taskpool或worker的消息通信机制将该对象共享到其他ArkTS线程。
生产者与消费者模式
相关文章:
鸿蒙开发中的并发与多线程
文章目录 前言异步并发 (Promise和async/await)多线程并发并发能力选择耗时任务并发执行场景常见业务场景 常驻任务并发执行场景常见业务场景 传统共享内存并发业务长时任务并发执行场景常见业务场景 并发任务管理线程间通信同语言线程间通信(ArkTS内)线…...
TCP和UDP的区别是什么?
1. 基本特性: TCP: 面向连接:在数据传输开始前,TCP需要在通信双方建立连接(三次握手)。可靠性:TCP保证数据的可靠传输,通过确认应答、重传机制、数据包顺序等确保数据无误到达。流量控制和拥塞…...
MySQL 函数(入门版)
目录 一、字符串函数 1、常用的字符串函数 2、函数演示 3、具体案例 二、数值函数 1、常用的数值函数 2、函数演示 3、具体案例 三、日期函数 1、常用的日期函数 2、函数演示 3、具体案例 四、流程函数 1、常用的流程函数 2、函数演示 3、具体案例 在MySQL中&a…...
Simulink中Signal Builder在新版中找不到怎么办
在较新的MATLAB版本中,新版Simulink中的Signal Builder用Signal Editor作为替代工具。 signal builder not shown in matlab - MATLAB Answers - MATLAB Central signalBuilderToSignalEditor 1.打开上面第二个链接 2.点击拷贝 3.然后在命令行中粘贴 4.然后就会…...
【补题】P10424 [蓝桥杯 2024 省 B] 好数(数位dp)
题意: 一个整数如果按从低位到高位的顺序,奇数位(个位、百位、万位……)上的数字是奇数,偶数位(十位、千位、十万位……)上的数字是偶数,我们就称之为“好数”。 给定一个正整数 N…...
SvelteKit 最新中文文档教程(19)—— 最佳实践之身份认证
前言 Svelte,一个语法简洁、入门容易,面向未来的前端框架。 从 Svelte 诞生之初,就备受开发者的喜爱,根据统计,从 2019 年到 2024 年,连续 6 年一直是开发者最感兴趣的前端框架 No.1: Svelte …...
Cursor编程-从入门到精通__0409
早期的Github Copilot 最近更新了,支持Agent编程,字节跳动Trae使用(免费),但成熟程度不如Cursor,Cursor前50次免费 Copilot VS Cursor*** 1,Cursor VSCode 二次开发,IDE级别 2&…...
VSCode、clangd、mingw 配置与使用
1.安装 安装如下软件: VSCodeclangd 扩展mingw-w64 2.配置 配置好 mingw-w64 到用户环境中。 在项目中设置 .clangd 扩展,设置 argument //setting.json"clangd.arguments": ["--query-driverD:\\Development\\Tools\\mingw64\\bin…...
深度学习处理文本(14)
使用Transformer进行序列到序列学习 正是序列到序列学习让Transformer真正大放异彩。与RNN相比,神经注意力使Transformer模型能够处理更长、更复杂的序列。要将英语翻译成西班牙语,你不会一个单词一个单词地阅读英语句子,将其含义保存在记忆中,然后再一个单词一个单词地生…...
核心案例 | 湖南汽车工程职业大学无人机操控与编队技术实验室
核心案例 | 湖南汽车工程职业大学无人机操控与编队技术实验室 为满足当今无人机行业应用需求,推动无人机技术的教育与实践深度融合,北京卓翼智能科技有限公司旗下品牌飞思实验室与湖南汽车工程职业大学强强联手,共同建设无人机操控与编队技术…...
Oracle 查看后台正在执行的 SQL 语句
在 Oracle 数据库中,要查看后台正在执行的 SQL 语句,可以通过查询动态性能视图(Dynamic Performance Views)或使用监控工具来实现。 1. 查询动态性能视图 (1) 查看当前活跃会话及其执行的 SQL 使用 v$session 和 v$sql 视图关联…...
SpringBoot整合MinIO快速入门:实现分布式文件存储与管理
文章目录 一、MinIO是什么?为什么选择它?1.1 什么是MinIO?1.2 核心优势 二、本地快速搭建MinIO服务2.1 Docker一键部署2.2 访问管理界面2.3 创建存储桶(Bucket) 三、SpringBoot集成MinIO客户端3.1 添加Maven依赖3.2 配…...
我的NISP二级之路-03
目录 一.ISMS 二.IP 三.http 四.防火墙 五.文件 解析 解析 六.攻击 解析 解析 七.风险管理工程 八.信息系统安全保护等级 九.我国信息安全保障 一.ISMS 1.文档体系建设是信息安全管理体系(ISMS)建设的直接体现,下列说法不正确的是: A&#…...
Vue框架的Diff算法
以下是关于 Diff 算法 的系统梳理: 一、Diff 算法的核心目标 最小化 DOM 操作:通过虚拟 DOM 对比,找出真实 DOM 的最小变更集高效节点复用:尽可能复用相同节点,减少创建/销毁开销顺序优化处理:优先处理高频变更场景(如列表尾部追加)保证渲染正确性:正确处理组件状态和…...
Oracle 表空间高水位收缩全攻略
1. 概述 本文档是针对某个特定用户表空间收缩的文档,实际操作要结合生产库具体情况。主要包括以下几个流程: 收集当前数据库相关信息降低数据库表高水位线Resize 收缩数据文件 具体细节详见以下章节。 2. 时间规划 操作类型预估时间实际时间数据库信…...
ESModule和CommonJS在Node中的区别
ESModule console.log(require);//>errorconsole.log(module);//>errorconsole.log(exports);//>errorconsole.log(__filename);//>errorconsole.log(__dirname);//>error全部报错commonjs console.log(require);console.log(module);console.log(exports);co…...
floyd模板
B3647 【模板】Floyd - 洛谷 f l o y d floyd floyd 模板 对于 f l o y d floyd floyd 算法来说时间复杂度为 O ( n 3 ) O(n^3) O(n3) ,不如跑 n n n 遍 h e a p _ d i j k s t r a heap\_dijkstra heap_dijkstra 算法 题目大意: 给出一张由 n n …...
力扣刷题-热题100题-第34题(c++、python)
23. 合并 K 个升序链表 - 力扣(LeetCode)https://leetcode.cn/problems/merge-k-sorted-lists/?envTypestudy-plan-v2&envIdtop-100-liked 顺序合并 合并两个有序链表作为子函数,创建一个空链表,然后对含有多个链表的数组进…...
括号匹配问题--栈
括号匹配问题 栈的应用代码概览栈操作函数详解1.初始化栈(stackInit)2.向栈中压入元素(stackpush)3.获取栈顶元素(stacktop)4.弹出栈顶元素(stackpop)5.销毁栈(stackdest…...
原生SSE实现AI智能问答+Vue3前端打字机流效果
实现流程: 1.用户点击按钮从右侧展开抽屉(drawer),打开模拟对话框 2.用户输入问题,点击提问按钮,创建一个SSE实例请求后端数据,由于SSE是单向流,所以每提一个问题都需要先把之前的实…...
LLC工作模态详解
1以半桥LLC谐振变换器为例,主开关Q1、Q2构成半桥结构,其驱动信号为固定占空比50%的互补信号,并且在上下桥臂之间应有死区时间。 谐振电感Ls、谐振电感Cs和变压器励磁电感Lm共同构成谐振槽路,具有两个谐振频率: 谐振电…...
线代第三课:n阶行列式
引言 行标取自然排列 不同行不同列的3个元素相乘 列标取排列的所有可能 列标排列的逆序数的奇偶性决定符号,- n阶行列式 第一种:按行展开 (1) 行标取自然排列 (2) 列标取排列的所有可能 (PS:可以理解为随意取) (3) 从…...
机器学习的一百个概念(10)假阳性率
前言 本文隶属于专栏《机器学习的一百个概念》,该专栏为笔者原创,引用请注明来源,不足和错误之处请在评论区帮忙指出,谢谢! 本专栏目录结构和参考文献请见[《机器学习的一百个概念》 ima 知识库 知识库广场搜索: 知识库创建人机器学习@Shockang机器学习数学基础@Shocka…...
GitHub 克隆/下载失败的解决方案
🚀 GitHub 下载/克隆失败?一招搞定代理配置与回滚! 在国内使用 Git 操作 GitHub 时,经常会遇到以下问题: ❌ 下载失败、超时 ❌ Failed to connect to github.com port 443 ❌ SSL certificate problem 本文将详细讲解…...
pulsar proxy详解
什么是 Pulsar Proxy? Pulsar Proxy 是 Apache Pulsar 中的一个可选组件,作用是作为客户端与 Pulsar Brokers 之间的中间网关层。它并不是 Pulsar 核心功能必须的部分,但在特定场景下(如复杂的网络环境、安全性需求或动态集群管理…...
C++ Socket优化实战:提升网络应用的性能与效率
🧑 博主简介:CSDN博客专家、CSDN平台优质创作者,高级开发工程师,数学专业,拥有高级工程师证书;擅长C/C、C#等开发语言,熟悉Java常用开发技术,能熟练应用常用数据库SQL server,Oracle…...
STM32单片机入门学习——第30节: [9-6] FlyMcu串口下载STLINK Utility
写这个文章是用来学习的,记录一下我的学习过程。希望我能一直坚持下去,我只是一个小白,只是想好好学习,我知道这会很难,但我还是想去做! 本文写于:2025.04.09 STM32开发板学习——第30节: [9-6] FlyMcu串口下载&STLINK Utility 前言开发…...
Qt容器类在元对象系统中使用
解释 “QVector没有被注册到Qt的元对象系统中”这句话的意思是:QVector<double>这种数据类型没有被Qt的元对象系统(Meta-Object System)识别和管理。Qt的元对象系统是Qt框架的核心部分,它提供了信号与槽机制、动态属性系统…...
亮相CMEF,美的医疗全维度打造智慧医疗新生态
当下,医疗科技革命的浪潮正汹涌而来,AI技术在中国医疗器械领域迅猛发展,释放出巨大的潜力。 4月8日,在第91届中国国际医疗器械博览会(CMEF)上,2025美的医疗年度新品发布暨中国脊梁守护计划启动…...
数据库视图讲解(view)
一、为什么需要视图 二、视图的讲解 三、总结 一、为什么需要视图 视图一方面可以帮我们使用表的一部分而不是所有的表,另一方面也可以针对不同的用户制定不同的查询视图。 比如,针对一个公司的销售人员,我们只想给他看部分数据,…...
