# 启动一个Worker > ![icon-note.gif](public_sys-resources/icon-note.gif) **说明:** > 本模块首批接口从API version 7开始支持。后续版本的新增接口,采用上角标单独标记接口的起始版本。 Worker是与主线程并行的独立线程。创建Worker的线程称之为宿主线程,Worker自身的线程称之为Worker线程。创建Worker传入的url文件在Worker线程中执行,可以处理耗时操作但不可以直接操作UI。 ## 导入模块 ```js import worker from '@ohos.worker'; ``` ## 属性 **系统能力:** 以下各项对应的系统能力均为SystemCapability.Utils.Lang。 | 名称 | 参数类型 | 可读 | 可写 | 说明 | | ---------- | --------------------------------------------------------- | ---- | ---- | ------------------------------------ | | parentPort | [DedicatedWorkerGlobalScope](#dedicatedworkerglobalscope) | 是 | 是 | worker线程用于与宿主线程通信的对象。 | ## WorkerOptions Worker构造函数的选项信息,用于为Worker添加其他信息。 **系统能力:** 以下各项对应的系统能力均为SystemCapability.Utils.Lang。 | 名称 | 参数类型 | 可读 | 可写 | 说明 | | ------ | --------- | ---- | ---- | ---------------------- | | name | string | 是 | 是 | Worker的名称。 | | shared | boolean | 是 | 是 | Worker是否可以被分享。 | ## Worker 使用以下方法前,均需先构造Worker实例,Worker类继承[EventTarget](#eventtarget)。 ### constructor constructor(scriptURL: string, options?: WorkerOptions) Worker构造函数。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | --------- | ------------------------------- | ---- | ------------------------------------------------------------ | | scriptURL | string | 是 | Worker执行脚本的url。
在FA和Stage模型下,DevEco Studio新建Worker工程路径分别存在以下两种情况:
(a) workers目录与pages目录同级。
(b) workers目录与pages目录不同级。 | options | [WorkerOptions](#workeroptions) | 否 | Worker构造的选项。 | **返回值:** | 类型 | 说明 | | ------ | --------------------------------------------------------- | | Worker | 执行Worker构造函数生成的Worker对象,失败则返回undefined。 | **示例:** ```js import worker from '@ohos.worker'; // worker线程创建 // FA模型-目录同级 const workerFAModel01 = new worker.Worker("workers/worker.js", {name:"first worker"}); // FA模型-目录不同级(以workers目录放置pages目录前一级为例) const workerFAModel02 = new worker.Worker("../workers/worker.js", {name:"first worker"}); // Stage模型-目录同级 const workerStageModel01 = new worker.Worker('entry/ets/workers/worker.ts'); // Stage模型-目录不同级(以workers目录放置pages目录后一级为例) const workerStageModel02 = new worker.Worker('entry/ets/pages/workers/worker.ts'); // scriptURL——"entry/ets/workers/worker.ts"的解释: // entry: 为module.json5中module中name属性的值; // ets: 表明当前使用的语言。 ``` 同时,需在工程目录下build-profile.json5文件的buildOption属性中添加配置信息,主要分为下面两种情况: (1) 目录同级( **不添加也可以** ) FA模型: ```json "buildOption": { "sourceOption": { "workers": [ "./src/main/ets/MainAbility/workers/worker.ts" ] } } ``` Stage模型: ```json "buildOption": { "sourceOption": { "workers": [ "./src/main/ets/workers/worker.ts" ] } } ``` (2) 目录不同级( **必须添加** ) FA模型: ```json "buildOption": { "sourceOption": { "workers": [ "./src/main/ets/workers/worker.ts" ] } } ``` Stage模型: ```json "buildOption": { "sourceOption": { "workers": [ "./src/main/ets/pages/workers/worker.ts" ] } } ``` ### postMessage postMessage(message: Object, options?: PostMessageOptions): void 向Worker线程发送消息,数据的传输采用结构化克隆算法。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------- | ----------------------------------------- | ---- | ------------------------------------------------------------ | | message | Object | 是 | 发送至Worker的数据。 | | options | [PostMessageOptions](#postmessageoptions) | 否 | 可转移对象是 ArrayBuffer 的实例对象。transferList数组中不可传入null。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.postMessage("hello world"); var buffer = new ArrayBuffer(8); workerInstance.postMessage(buffer, [buffer]); ``` ### on on(type: string, listener: EventListener): void 向Worker添加一个事件监听。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | -------- | ------------------------------- | ---- | ---------------- | | type | string | 是 | 监听事件的type。 | | listener | [EventListener](#eventlistener) | 是 | 回调的事件。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.on("alert", (e)=>{ console.log("alert listener callback"); }) ``` ### once once(type: string, listener: EventListener): void 向Worker添加一个事件监听,事件监听只执行一次便自动删除。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | -------- | ------------------------------- | ---- | ---------------- | | type | string | 是 | 监听事件的type。 | | listener | [EventListener](#eventlistener) | 是 | 回调的事件。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.once("alert", (e)=>{ console.log("alert listener callback"); }) ``` ### off off(type: string, listener?: EventListener): void 删除Worker的事件监听。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | -------- | ------------------------------- | ---- | ---------------------- | | type | string | 是 | 需要删除事件的type。 | | listener | [EventListener](#eventlistener) | 否 | 需要删除的回调的事件。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.off("alert"); ``` ### terminate terminate(): void 关闭Worker线程,终止Worker接收消息。 **系统能力:** SystemCapability.Utils.Lang **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.terminate(); ``` ### onexit onexit?: (code: number) => void Worker对象的onexit属性表示Worker退出时被调用的事件处理程序,处理程序在宿主线程中执行。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------ | ---- | ------------------ | | code | number | 否 | Worker退出的code。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.onexit = function(e) { console.log("onexit"); } ``` ### onerror onerror?: (err: ErrorEvent) => void Worker对象的onerror属性表示Worker在执行过程中发生异常被调用的事件处理程序,处理程序在宿主线程中执行。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------------------------- | ---- | ---------- | | err | [ErrorEvent](#errorevent) | 否 | 异常数据。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.onerror = function(e) { console.log("onerror"); } ``` ### onmessage onmessage?: (event: MessageEvent\) => void Worker对象的onmessage属性表示宿主线程接收到来自其创建的Worker通过parentPort.postMessage接口发送的消息时被调用的事件处理程序,处理程序在宿主线程中执行。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ----------------------------- | ---- | ---------------------- | | event | [MessageEvent](#messageevent) | 否 | 收到的Worker消息数据。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.onmessage = function(e) { // e : MessageEvent, 用法如下: // let data = e.data; console.log("onmessage"); } ``` ### onmessageerror onmessageerror?: (event: MessageEvent\) => void Worker对象的onmessageerror属性表示当Worker对象接收到一条无法被序列化的消息时被调用的事件处理程序,处理程序在宿主线程中执行。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ----------------------------- | ---- | ---------- | | event | [MessageEvent](#messageevent) | 否 | 异常数据。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.onmessageerror= function(e) { console.log("onmessageerror"); } ``` ## EventTarget ### addEventListener addEventListener(type: string, listener: EventListener): void 向Worker添加一个事件监听。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | -------- | ------------------------------- | ---- | ---------------- | | type | string | 是 | 监听事件的type。 | | listener | [EventListener](#eventlistener) | 是 | 回调的事件。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.addEventListener("alert", (e)=>{ console.log("alert listener callback"); }) ``` ### removeEventListener removeEventListener(type: string, callback?: EventListener): void 删除Worker的事件监听。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | -------- | ------------------------------- | ---- | ---------------------- | | type | string | 是 | 需要删除事件的type。 | | callback | [EventListener](#eventlistener) | 否 | 需要删除的回调的事件。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.removeEventListener("alert"); ``` ### dispatchEvent dispatchEvent(event: Event): boolean 分发定义在Worker的事件。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | --------------- | ---- | ---------------- | | event | [Event](#event) | 是 | 需要分发的事件。 | **返回值:** | 类型 | 说明 | | ------- | ------------------------------- | | boolean | 分发的结果,false表示分发失败。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.dispatchEvent({type:"alert"}); ``` ### removeAllListener removeAllListener(): void 删除Worker的所有事件监听。 **系统能力:** SystemCapability.Utils.Lang **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.removeAllListener(); ``` ## DedicatedWorkerGlobalScope Worker线程用于与宿主线程通信的类,通过postMessage接口发送消息给宿主线程、close接口关闭Worker线程,DedicatedWorkerGlobalScope类继承[WorkerGlobalScope](#workerglobalscope)。 ### postMessage postMessage(messageObject: Object, options?: PostMessageOptions): void Worker向宿主线程发送消息。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------- | ----------------------------------------- | ---- | ------------------------------------------------------------ | | message | Object | 是 | 发送至Worker的数据。 | | options | [PostMessageOptions](#postmessageoptions) | 否 | 可转移对象是ArrayBuffer的实例对象。transferList数组中不可传入null。 | **示例:** ```js // main.js import worker from '@ohos.worker'; const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.postMessage("hello world"); workerInstance.onmessage = function(e) { // let data = e.data; console.log("receive data from worker.js"); } ``` ```js // worker.js import worker from '@ohos.worker'; const parentPort = worker.parentPort; parentPort.onmessage = function(e){ // let data = e.data; parentPort.postMessage("receive data from main.js"); } ``` ### close close(): void 关闭Worker线程,终止Worker接收消息。 **系统能力:** SystemCapability.Utils.Lang **示例:** ```js // main.js import worker from '@ohos.worker'; const workerInstance = new worker.Worker("workers/worker.js"); ``` ```js // worker.js import worker from '@ohos.worker'; const parentPort = worker.parentPort; parentPort.onmessage = function(e) { parentPort.close() } ``` ### onmessage onmessage?: (event: MessageEvent\) => void DedicatedWorkerGlobalScope的onmessage属性表示Worker线程收到来自其宿主线程通过worker.postMessage接口发送的消息时被调用的事件处理程序,处理程序在Worker线程中执行。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ----------------------------- | ---- | ---------------------- | | event | [MessageEvent](#messageevent) | 否 | 收到的Worker消息数据。 | **示例:** ```js // main.js import worker from '@ohos.worker'; const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.postMessage("hello world"); ``` ```js // worker.js import worker from '@ohos.worker'; const parentPort = worker.parentPort; parentPort.onmessage = function(e) { console.log("receive main.js message"); } ``` ### onmessageerror onmessageerror?: (event: MessageEvent\) => void DedicatedWorkerGlobalScope的onmessageerror属性表示当Worker对象接收到一条无法被反序列化的消息时被调用的事件处理程序,处理程序在Worker线程中执行。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ----------------------------- | ---- | ---------- | | event | [MessageEvent](#messageevent) | 否 | 异常数据。 | **示例:** ```js // main.js import worker from '@ohos.worker'; const workerInstance = new worker.Worker("workers/worker.js"); ``` ```js // worker.js import worker from '@ohos.worker'; const parentPort = worker.parentPort; parentPort.onmessageerror= function(e) { console.log("worker.js onmessageerror") } ``` ## PostMessageOptions 明确数据传递过程中需要转移所有权对象的类,传递所有权的对象必须是ArrayBuffer。 **系统能力:** 以下各项对应的系统能力均为SystemCapability.Utils.Lang。 | 名称 | 参数类型 | 可读 | 可写 | 说明 | | -------- | -------- | ---- | ---- | --------------------------------- | | transfer | Object[] | 是 | 是 | ArrayBuffer数组,用于传递所有权。 | ## Event 事件类。 **系统能力:** 以下各项对应的系统能力均为SystemCapability.Utils.Lang。 | 名称 | 参数类型 | 可读 | 可写 | 说明 | | --------- | -------- | ---- | ---- | ---------------------------------- | | type | string | 是 | 否 | 指定事件的type。 | | timeStamp | number | 是 | 否 | 事件创建时的时间戳(精度为毫秒)。 | ## EventListener 事件监听类。 ### (evt: Event): void | Promise<void> 执行的回调函数。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | --------------- | ---- | -------------- | | evt | [Event](#event) | 是 | 回调的事件类。 | **返回值:** | 类型 | 说明 | | ------------------------------------- | ------------------------------- | | void \| Promise<void> | 无返回值或者以Promise形式返回。 | **示例:** ```js const workerInstance = new worker.Worker("workers/worker.js"); workerInstance.addEventListener("alert", (e)=>{ console.log("alert listener callback"); }) ``` ## ErrorEvent 错误事件类,用于表示Worker执行过程中出现异常的详细信息,ErrorEvent类继承[Event](#event)。 **系统能力:** 以下各项对应的系统能力均为SystemCapability.Utils.Lang。 | 名称 | 参数类型 | 可读 | 可写 | 说明 | | -------- | -------- | ---- | ---- | -------------------- | | message | string | 是 | 否 | 异常发生的错误信息。 | | filename | string | 是 | 否 | 出现异常所在的文件。 | | lineno | number | 是 | 否 | 异常所在的行数。 | | colno | number | 是 | 否 | 异常所在的列数。 | | error | Object | 是 | 否 | 异常类型。 | ## MessageEvent 消息类,持有Worker线程间传递的数据。 **系统能力:** 以下各项对应的系统能力均为SystemCapability.Utils.Lang。 | 名称 | 参数类型 | 可读 | 可写 | 说明 | | ---- | -------- | ---- | ---- | ------------------ | | data | T | 是 | 否 | 线程间传递的数据。 | ## WorkerGlobalScope Worker线程自身的运行环境,WorkerGlobalScope类继承[EventTarget](#eventtarget)。 ### 属性 **系统能力:** 以下各项对应的系统能力均为SystemCapability.Utils.Lang。 | 名称 | 参数类型 | 可读 | 可写 | 说明 | | ---- | ------------------------------------------------------------ | ---- | ---- | --------------------------------------- | | name | string | 是 | 否 | Worker的名字,有new Worker时指定。 | | self | [WorkerGlobalScope](#workerglobalscope) & typeof globalThis | 是 | 否 | WorkerGlobalScope本身。 | ### onerror onerror?: (ev: ErrorEvent) => void WorkerGlobalScope的onerror属性表示Worker在执行过程中发生异常被调用的事件处理程序,处理程序在Worker线程中执行。 **系统能力:** SystemCapability.Utils.Lang **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------------------------- | ---- | ---------- | | ev | [ErrorEvent](#errorevent) | 否 | 异常数据。 | **示例:** ```js // main.js import worker from '@ohos.worker'; const workerInstance = new worker.Worker("workers/worker.js") ``` ```js // worker.js import worker from '@ohos.worker'; const parentPort = worker.parentPort parentPort.onerror = function(e){ console.log("worker.js onerror") } ``` ## 其他说明 ### 内存模型 Worker基于Actor并发模型实现。在Worker的交互流程中,JS主线程可以创建多个Worker子线程,各个Worker线程间相互隔离,并通过序列化传递对象,等到Worker线程完成计算任务,再把结果返回给主线程。 Actor并发模型的交互原理:各个Actor并发地处理主线程任务,每个Actor内部都有一个消息队列及单线程执行模块,消息队列负责接收主线程及其他Actor的请求,单线程执行模块则负责串行地处理请求、向其他Actor发送请求以及创建新的Actor。由于Actor采用的是异步方式,各个Actor之间相互隔离没有数据竞争,因此Actor可以高并发运行。 ### 注意事项 - Worker存在数量限制,当前支持最多同时存在7个Worker。 - 当Worker数量超出限制,会出现Error "Too many workers, the number of workers exceeds the maximum."。 - 主动销毁Worker可以调用新创建Worker对象的terminate()或parentPort.close()方法。 - Worker的创建和销毁耗费性能,建议管理已创建的Worker并重复使用。 ## 完整示例 ### FA模型 ```js // main.js(同级目录为例) import worker from '@ohos.worker'; // 主线程中创建Worker对象 const workerInstance = new worker.Worker("workers/worker.ts"); // 创建js和ts文件都可以 // const workerInstance = new worker.Worker("workers/worker.js"); // 主线程向worker线程传递信息 workerInstance.postMessage("123"); // 主线程接收worker线程信息 workerInstance.onmessage = function(e) { // data:worker线程发送的信息 let data = e.data; console.log("main.js onmessage"); // 销毁Worker对象 workerInstance.terminate(); } // 在调用terminate后,执行回调onexit workerInstance.onexit = function() { console.log("main.js terminate"); } ``` ```js // worker.js import worker from '@ohos.worker'; // 创建worker线程中与主线程通信的对象 const parentPort = worker.parentPort // worker线程接收主线程信息 parentPort.onmessage = function(e) { // data:主线程发送的信息 let data = e.data; console.log("worker.js onmessage"); // worker线程向主线程发送信息 parentPort.postMessage("123") } // worker线程发生error的回调 parentPort.onerror= function(e) { console.log("worker.js onerror"); } ``` build-profile.json5 配置 : ```json "buildOption": { "sourceOption": { "workers": [ "./src/main/ets/MainAbility/workers/worker.ts" ] } } ``` ### Stage模型 ```js // main.js(以不同目录为例) import worker from '@ohos.worker'; // 主线程中创建Worker对象 const workerInstance = new worker.Worker("entry/ets/pages/workers/worker.ts"); // 创建js和ts文件都可以 // const workerInstance = new worker.Worker("entry/ets/pages/workers/worker.js"); // 主线程向worker线程传递信息 workerInstance.postMessage("123"); // 主线程接收worker线程信息 workerInstance.onmessage = function(e) { // data:worker线程发送的信息 let data = e.data; console.log("main.js onmessage"); // 销毁Worker对象 workerInstance.terminate(); } // 在调用terminate后,执行onexit workerInstance.onexit = function() { console.log("main.js terminate"); } ``` ```js // worker.js import worker from '@ohos.worker'; // 创建worker线程中与主线程通信的对象 const parentPort = worker.parentPort // worker线程接收主线程信息 parentPort.onmessage = function(e) { // data:主线程发送的信息 let data = e.data; console.log("worker.js onmessage"); // worker线程向主线程发送信息 parentPort.postMessage("123") } // worker线程发生error的回调 parentPort.onerror= function(e) { console.log("worker.js onerror"); } ``` build-profile.json5 配置: ```json "buildOption": { "sourceOption": { "workers": [ "./src/main/ets/pages/workers/worker.ts" ] } } ```