ArtTS语言基础类库内容的学习(2.10.1)

上篇回顾: ArkTS开发系列之Web组件的学习(2.9)

本篇内容:ArtTS语言基础类库-异步同步并发内容的学习(2.10.1)

一、知识储备

1. 异常并发

Promise和Async/await提供异步并发能力,是标准的JS异步语法

  • Promise是一种用于处理异常操作的对象, 可以将异步操作转换为类似于同步操作的风格,以方便代码编写和维护。
  • 分别有pending(进行中)、fulfilled(已完成)和rejected(已拒绝)。
  • 下面是个例子
  promiseFunction() {

    const promise = new Promise((resolve, reject) => {
      setTimeout(() => {//通过 setTimeout来模拟一个异步操作   此时处于进行中状态(pending)
        const randomNumber = Math.random();
        if (randomNumber > 0.5) {//随机数大于0.5, 则正常返回,否则返回异常信息
          resolve(randomNumber)
        } else {
          reject(new Error('Random number is too small'))
        }
      }, 1000)
    })

    promise.then(result => {// fulfilled 成功完成回调状态
      this.tips = "异步结果:" + result;
      promptAction.showToast({ message: this.tips });
    }).catch(err => {// rejected 已被拒绝的回调状态
      this.tips = '异步结果:' + err.message
      promptAction.showToast({ message: this.tips });
    })
  }

async/await是一种用于处理异步操作的Promise语法糖, 通过使用async关键字来声明一个函数为异步函数, 并使用await关键字等promise的成功或拒绝回调。以同步的方式来编写异步代码,使代码更为简洁、易读。

  async asyncPromiseFunction() {
    const result = await new Promise((resolve, reject) => {
      setTimeout(() => {
        resolve('success')
      }, 1000)
    })
    console.log('结果:' + result)
    this.tips = "异步结果:" + result;
    promptAction.showToast({ message: this.tips });
  }

2.1 多线程并发

常见的并发模型分为基于内存共享并发模型和基于消息通信并发模型两种。
Actor并发模型作为基于消息通信并发模型的典型代表,不需要开发者去面对锁带来的一系列复杂偶发的问题,同时并发度也相对较高。
由于Actor模型的内存隔离特性,所以需要进行跨线程的数据序列化传输。

  • 数据传输对象分为普通对象、可转移对象、可共享对象、Native绑定对象四种
  • 普通对象采用结构化克隆算法(Structured Clone)进行序列化传输,此算法可以通过递归的方式拷贝传输对象,相较所支持的对象类型更加丰富。
    • 可传输对象包括除Symbol之外的基础类型、Date、String、RegExp、Array、Map、Set、ArrayBuffer、TypedArray、Object(仅限简单对象,比如通过"{}"或“new object”创建。仅支持传递属性、不支持传递其原型及方法)
  • 可转移对象采用地址转移进行序列化传输,不需要内容拷贝,会将ArrayBuffer的所有权转移给接收该ArrayBuffer的线程,转移后该ArrayBuffer在发送它的线程中变为不可用, 不允许再访问。
    • 定义可转移对象
    let buffer = new ArrayBuffer(100);
    
  • 可共享对象指支持在多线程之间传递SharedArrayBuffer对象。传递之后的SharedArrayBuffer对象和原始的可以指向同一块内存,进而达到内存共享的目的。
    • SharedArrayBuffer对象存储的数据在同时被修改时,需要通过原子操作保证其同步性,即下个操作开始之前,务必需要等到上个操作已经结束。
    • 定义可共享对象
    let sharedBuffer = new SharedArrayBuffer(1024);
    
  • Native绑定对象(Native Binding Object),是系统所提供的对象,该对象与底层系统功能进行绑定,提供直接访问底层系统功能的能力。
    • 当前支持序列化传输的Native绑定对象主要包含:ContextRemoteObject.
    • Context对象包含应用程序组件的上下文信息,并提供了一种访问系统服务和资源的方式,方便开发者开发的应用与系统进行交互。
    • RemoteObject对象的主要作用是实现远程通信的功能,它允许在不同的进程间传递对象的引用 。使得不同进程之间可以共享对象的状态和方法,服务提供者必须继承此类。

2.2 TaskPool(任务池)和Worker

实现特点及使用场景直观对比

实现TaskPoolWorker差异点
内存模型线程间隔离,内存不共享线程间隔离,内存不共享相同
参数传递机制采用标准的结构化克隆算法(Structured Clone)进行序列化、反序列化,完成参数传递。支持ArrayBuffer转移和SharedArrayBuffer共享。采用标准的结构化克隆算法(Structured Clone)进行序列化、反序列化,完成参数传递。支持ArrayBuffer转移和SharedArrayBuffer共享。相同
参数传递直接传递,无需封闭,默认进行transfer。消息对象唯一参数,需要自己封装。是否需要封装不同
返回值异步调用后默认返回主动发送消息,需要在onMessage解析赋值接收返回值方式不同
生命周期TaskPool自动管理生命周期,无需关心任务负载高低问题开发者自动管理Worker的数量及生命周期是否需要开发者管理不同
任务池个数上限自动管理,无需配置同个进程下,最多支持同时开启8个Worker线程开启线程个数不同
任务执行时长上限无限制无限制相同
设备任务的优先级不支持不支持相同
执行任务的取消支持取消任务队列中等待的任务不支持Worker不支持取消任务
适用场景需要频繁取消的任务,例如图库大图浏览场景,为提升体验,会同时缓存当前图片左右侧各2张图片,适合使用TaskPool;大师或者调度点分散的任务,例如大型应用的多个模块包含多个耗时任务,不方便使用8个Worker去做负载管理。推荐使用TaskPool有关联的一系列同步任务,例如每次创建、使用不同句柄,且句柄需要永久保存,保证使用该句柄操作,需要使用Worker。
  • TaskPool 注意事项
    • 实现任务的函数需要使用装饰器@Concurrent标注,且仅支持在.ets文件中使用
    • 实现任务的函数入参,需满足序列化支持的类型,
    • 由于线程中上下文对象是不同的,因此TaskPool工作线程中只能使用线程安全的库。不能做ui相关操作,
    • 序列化传输的数据量大小限制为16MB
  • Worker 注意事项
    • 创建Worker时,传入的Worker.ts路径在不同版本有不同的规则
    • Worker创建后需要手动管理生命周期,且同进程中最多同时运行的Worker子线程数量为8个
    • Ability类型的Modul支持使用Worker,Library类型的Module不支持使用Worker
    • 创建Worker不支持使用其他Module的Wroker.ts文件,即不支持跨模块调用Worker
    • 由于不同线程中上下文对象是不同的,因此Worker线程只能使用线程安全的库,不能做ui相关操作。
    • 序列化传输的数据量大小限制为16MB
  • 文件路径注意事项
    • 创建Worker时,需要传入Worker的路径,Worker文件存放位置默认路径为Worker文件所在目录与pages目录同级
      //导入模块
      import worker from '@ohos.worker';
      
      //APi8及之前
      const worker = new worker.Worker(scriptUrl);
      
      //APi 9及之后
      const worker = new worker.ThreadWorker(scriptUrl);
      //写法一
      const worker = new worker.ThreadWorker('entry/ets/workers/MyWorker.ts', {name: "name"});
      //写法二
      const worker = new worker.ThreadWorker('@bundle:com.aji.study/entry/ets/workers/worker')
      

2.3 @Concurrent装饰器

使用TaskPool时,执行的头发函数需要使用该装饰器修饰,否则无法通过相关校验。
注意

  • 仅支持在Stage模型且module的compileMode为esmodule的project中使用taskpool api。确认module的compileMode方法:查看当前module的build-profile.json5,在buildOption中补充"compileMode": “esmodule”。
    @Concurrent
    function add(numa: number, numb: number): number { //声明一个加法函数
      return numa + numb;
    }
    
    async function calcTotal(): Promise<void> {
      try {
        let task: taskpool.Task = new taskpool.Task(add, 1, 2) //创建一个任务
        let count = await taskpool.execute(task);
        console.error("taskpool result is: " + count) //任务池执行任务
      } catch (err) {
        console.error("taskpool err is " + err.message);
      }
    }
    

二、效果一览

在这里插入图片描述

三、源码剖析

import promptAction from '@ohos.promptAction';
import fs from '@ohos.file.fs';
import common from '@ohos.app.ability.common';
import taskpool from '@ohos.taskpool';
import Handle from './Handle';
import worker from '@ohos.worker';

function promiseFunction() {

  const promise = new Promise((resolve, reject) => {
    setTimeout(() => { //通过 setTimeout来模拟一个异步操作   此时处于进行中状态(pending)
      const randomNumber = Math.random();
      if (randomNumber > 0.5) { //随机数大于0.5, 则正常返回,否则返回异常信息
        resolve(randomNumber)
      } else {
        reject(new Error('Random number is too small'))
      }
    }, 1000)
  })

  promise.then(result => { // fulfilled 成功完成回调状态
    let tips = "异步结果:" + result;
    promptAction.showToast({ message: tips });
  }).catch(err => { // rejected 已被拒绝的回调状态
    let tips = '异步结果:' + err.message
    promptAction.showToast({ message: tips });
  })
}

async function asyncPromiseFunction() {
  const result = await new Promise((resolve, reject) => {
    setTimeout(() => {
      resolve('success')
    }, 1000)
  })
  console.log('结果:' + result)
  let tips = "异步结果:" + result;
  promptAction.showToast({ message: tips });
}

async function write(data: string, file: fs.File): Promise<void> { //定义的I/O任务逻辑
  fs.write(file.fd, data).then((writeLen: number) => {
    console.info('write data length is : ' + writeLen); //写入的文本长度
  }).catch(err => {
    console.error(`failed code: ${err.code}, msg: ${err.message}`) //异常捕获 并输出日志
  })
}

async function writeFunction(): Promise<void> {
  let context = getContext() as common.UIAbilityContext;
  let filePath: string = context.filesDir + "/test.txt";

  let file: fs.File = await fs.open(filePath, fs.OpenMode.READ_WRITE | fs.OpenMode.CREATE)
  write('我要学鸿蒙', file).then(() => {
    console.info("write success")
    fs.close(file);
  }).catch(err => {
    console.error(`write failed code: ${err.code}, msg: ${err.message}`) //异常捕获 并输出日志
    fs.close(file);
  })
}

@Concurrent
function add(numa: number, numb: number): number { //声明一个加法函数
  return numa + numb;
}

async function calcTotal(): Promise<void> {
  try {
    let task: taskpool.Task = new taskpool.Task(add, 1, 2) //创建一个任务
    let count = await taskpool.execute(task);
    console.error("taskpool result is: " + count) //任务池执行任务
  } catch (err) {
    console.error("taskpool err is " + err.message);
  }
}

@Concurrent //步骤1 定义并发函数,内部调用同步方法
function age(num: number): boolean {
  Handle.syncSet(num);
  return true;
}

async function asyncGet() { //步骤2 创建任务并执行
  let task = new taskpool.Task(age, 3);
  let result = await taskpool.execute(task);
  console.error(`age is : ${result}`)
}

@Entry
@Component
struct Index { //异步和并发

  @State tips: string = '异步结果'

  build() {
    Column() {
      Button('Promise异步')
        .onClick(() => {
          promiseFunction();
        })
        .margin(20)

      Button('async异步')
        .onClick(() => {
          asyncPromiseFunction();
        })
        .margin(20)

      Button('I/O')
        .onClick(() => {
          writeFunction();
        })
        .margin(20)

      Button('年龄')
        .onClick(() => {
          calcTotal();
          asyncGet(); //执行并发操作
        })
        .margin(20)

      Button('worker')
        .onClick(() => {
          // let w = new worker.ThreadWorker('@bundle:com.aji.first/entry/ets/workers/MyWorker');
          let w = new worker.ThreadWorker('entry/ets/workers/MyWorker.ts');
          w.onmessage = function (msg) {
            console.error('msg: ' + JSON.stringify(msg))
          }
          w.onerror = function (err) {
            console.error("msg err:" + JSON.stringify(err))
          }

          w.postMessage({ 'type': 0, 'age': 19 })
          w.postMessage({ 'type': 1 })

          setTimeout(() => {
            w.terminate();
            console.error('msg destroy')
          }, 1000)
        })

    }.width('100%')
    .height('100%')
  }
}

import worker, { ThreadWorkerGlobalScope } from '@ohos.worker';
import Handle from '../pages/base/Handle';

var workerPort: ThreadWorkerGlobalScope = worker.workerPort;

var handler = new Handle();

workerPort.onmessage = function (msg) {
  console.error('msg workerPort: ' + JSON.stringify(msg))
  switch (msg.data.type) {
    case 0:
      Handle.syncSet(msg.data.age);
      workerPort.postMessage('success set')
      break;
    case 1:
      Handle.syncGet();
      workerPort.postMessage('success get')
      break;
  }
}

workerPort.onerror = function onErr(err) {

}
export default class Handle {
  static age: number;

  static instance: Handle = new Handle();

  constructor() {
  }

  static getInstance() {
    return this.instance;
  }

  static syncGet(): number {

    return this.age;
  }

  static syncSet(num: number) {
    this.age = num;
    return;
  }
}

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

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

相关文章

【王佩丰 Excel 基础教程】第一讲:认识Excel

文章目录 前言一、Excel软件简介1.1、历史上的其他数据处理软件与 Microsoft Excel1.2、Microsoft Excel 能做些什么1.3、Excel 界面介绍 二、Microsoft Excel 的一些重要概念2.1、Microsoft Excel 的几种常见文件类型2.2、工作簿、工作表、单元格. 三、使用小工具&#xff1a;…

基于Spring Boot的药房信息管理系统

1 项目介绍 1.1 研究的背景及意义 随着社会的飞速进步和药房行业竞争的白热化&#xff0c;传统的手工管理模式已难以适应药房信息管理的现代化需求。在计算机科学技术日臻完善的背景下&#xff0c;药房信息管理者们日益认识到运用计算机技术进行信息管理的迫切性和重要性。计…

黑马苍穹外卖8 Spring Task+WebSocket 来单提醒和客户催单

Spring Task Spring提供的任务调度工具&#xff0c;按照约定时间自动执行代码。【以前的都是基于请求(http)响应的】 cron表达式 通过cron表达式可以定义任务触发时间。 cron表达式生成器 &#xff08;1&#xff09;导入spring-context &#xff08;2&#xff09;EnableSc…

[OtterCTF 2018]Graphic‘s For The Weak

恶意软件的图形中有些可疑之处。 软件图形 &#xff1f;&#xff1f;&#xff1f;这里的恶意文件都是 vmware-tray.ex使用procdump转存进程的可执行文件 &#xff08;可执行的&#xff09;导出了 &#xff0c;看文件里面是否存在 图片 volatility.exe -f .\OtterCTF.vmem --pro…

Day38:LeedCode 1049. 最后一块石头的重量 II 494. 目标和 474.一和零

1049. 最后一块石头的重量 II 有一堆石头&#xff0c;用整数数组 stones 表示。其中 stones[i] 表示第 i 块石头的重量。 每一回合&#xff0c;从中选出任意两块石头&#xff0c;然后将它们一起粉碎。假设石头的重量分别为 x 和 y&#xff0c;且 x < y。那么粉碎的可能结果…

Linux - 记一次某Java程序启动报错(申请内存失败)

文章目录 问题可能原因分析可能原因分析尝试各种解决方案尝试解决过程 解决办法&#xff1a; 调整 overcommit_meory参数overcommit_memory详解什么是 overcommit_memory&#xff1f;overcommit_memory 的选项及其含义配置 overcommit_memory查看当前设置设置 overcommit_memor…

# linux 系统中,使用 “ ll “ 命令报错 “ bash ll command not found “ 解决方法:

linux 系统中&#xff0c;使用 " ll " 命令报错 " bash ll command not found " 解决方法&#xff1a; 一、错误描述&#xff1a; 报错原因&#xff1a; 1、这个错误表明你尝试在 bash shell 中执行 ll 命令&#xff0c;但是系统找不到这个命令。ll 通常…

Nvidia显卡GeForce Experience录屏操作流程

安装软件 首先我们从英伟达官网下载GeForce Experience程序&#xff0c;安装在电脑中GeForce Experience&#xff08;简称 GFE&#xff09;自动更新驱动并优化游戏设置 | NVIDIA 登录软件 安装完成后登录 开启录屏功能 登录后点击右上角的设置&#xff08;小齿轮图标&#x…

接口自动化测试关联token的方法?

引言&#xff1a; 在接口自动化测试中&#xff0c;有时候我们需要关联token来进行身份验证或权限管理。本文将从零开始&#xff0c;介绍如何详细且规范地实现接口自动化测试中token的关联。 步骤一&#xff1a;准备工作 在开始之前&#xff0c;我们需要确保以下准备工作已完成…

UE5(c++)demo开发日志(1):Actor类添加

工具---新建c类---选用Actor&#xff0c;创造出头文件(.h&#xff0c;用于声明变量、函数、重写父类函数等)和源文件(.cpp&#xff0c;涉及到具体功能实现)&#xff0c;客户端界面成功显示出来新建类&#xff1a; Actor头文件默认方法简介&#xff1a; UCLASS(): ue提供的宏&a…

前端工程化08-新的包管理工具pnpm

1、历史原因解读 pnpm这个东西发布的时间是比较早的&#xff0c;但是在最近一两年的时候才开始流行&#xff0c;甚至是可以说非常的盛行&#xff0c;那么这个包到底是个什么东西的&#xff0c;那么我们先说下&#xff0c;原来的包管理工具到底有那些问题&#xff1f;比如说我们…

第三阶段Spark

Spark和PySpark的介绍 PySpark的相关设置 安装PySpark库 pip install pyspark pip install -i https://pypi.tuna.tsinghua.edu.cn/simple pyspark 构建PySpark执行环境入口对象 # 导包 from pyspark import SparkConf, SparkContext# 创建SparkConf类对象 conf SparkConf()…

C++进修——C++核心编程

内存分区模型 C程序在执行时&#xff0c;将内存大方向划分为4个区域 代码区&#xff1a;存放函数体的二进制编码&#xff0c;由操作系统进行管理全局区&#xff1a;存放全局变量和静态变量以及常量栈区&#xff1a;由编译器自动分配释放&#xff0c;存放函数的参数值&#xff…

谈一下MySQL的两阶段提交机制

文章目录 为什么需要两阶段提交&#xff1f;两阶段提交流程&#xff1f;两阶段提交缺点&#xff1f; 为什么需要两阶段提交&#xff1f; 为了保证事务的持久性和一致性&#xff0c;MySQL需要确保redo log和binlog的同步持久化。MySQL通过“两阶段提交”的机制来实现在事务提交…

Python 面试【初级】

欢迎莅临我的博客 &#x1f49d;&#x1f49d;&#x1f49d;&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」…

安卓开发自定义时间日期显示组件

安卓开发自定义时间日期显示组件 问题背景 实现时间和日期显示&#xff0c;左对齐和对齐两种效果&#xff0c;如下图所示&#xff1a; 问题分析 自定义view实现一般思路&#xff1a; &#xff08;1&#xff09;自定义一个View &#xff08;2&#xff09;编写values/attrs.…

day57---面试专题(框架篇)

框架篇 1. Spring refresh 流程 要求 掌握 refresh 的 12 个步骤Spring refresh 概述 refresh 是 AbstractApplicationContext 中的一个方法,负责初始化 ApplicationContext 容器,容器必须调用 refresh 才能正常工作。它的内部主要会调用 12 个方法,我们把它们称为 refre…

【JavaScript】JavaScript简介

希望文章能给到你启发和灵感&#xff5e; 如果觉得文章对你有帮助的话&#xff0c;点赞 关注 收藏 支持一下博主吧&#xff5e; 阅读指南 JavaScript入门&#xff08;1&#xff09;————JavaScript简介开篇说明一、什么是JavaScript二、JavaScript的使用2.1 开发工具的选择…

上市公司环境研究汇总数据集(2008-2022年)

数据简介&#xff1a;上市公司环境研究是指对上市公司在环境保护和可持续发展方面的表现和做法进行评估和研究。这些评估可以包括上市公司的环境风险、环境管理制度和政策、环境负债和环境绩效等方面。 上市公司环境研究可以帮助上市公司更好地了解自身的环境状况和风险&#…

FastAPI教程II

本文参考FastAPI教程https://fastapi.tiangolo.com/zh/tutorial Cookie参数 定义Cookie参数与定义Query和Path参数一样。 具体步骤如下&#xff1a; 导入Cookie&#xff1a;from fastapi import Cookie声明Cookie参数&#xff0c;声明Cookie参数的方式与声明Query和Path参数…