Workers

Deno支持 Web Worker API.

Workers可以被用来在多线程上执行代码。每个 Worker 实例都运行在单独的线程上,每个线程只专用于这个worker。

到目前为止,Deno只支持 module 类型的workers,因此,创建新worker的时候,传递 type: "module" 就是必要的了:

  1. // Good
  2. new Worker("./worker.js", { type: "module" });
  3. // Bad
  4. new Worker("./worker.js");
  5. new Worker("./worker.js", { type: "classic" });

权限

创建一个新的 Worker 实例和动态引入类似;因此,Deno要求适当的权限来执行这个动作。

对于使用本地模块的workers,allow-read 权限时必需的:

  1. // main.ts
  2. new Worker("./worker.ts", { type: "module" });
  3. // worker.ts
  4. console.log("hello world");
  5. self.close();
  1. $ deno run main.ts
  2. error: Uncaught PermissionDenied: read access to "./worker.ts", run again with the --allow-read flag
  3. $ deno run --allow-read main.ts
  4. hello world

对于使用远程模块的workers,--allow-net 权限是必需的:

  1. // main.ts
  2. new Worker("https://example.com/worker.ts", { type: "module" });
  3. // worker.ts
  4. console.log("hello world");
  5. self.close();
  1. $ deno run main.ts
  2. error: Uncaught PermissionDenied: net access to "https://example.com/worker.ts", run again with the --allow-net flag
  3. $ deno run --allow-net main.ts
  4. hello world

在worker中使用Deno

这是一个不稳定的特性。详情请参考 unstable features

默认情况下,在worker中 Deno 命名空间并不适用。

为了添加 Deno 命名空间,在创建worker时传入 deno: true 选项:

  1. // main.js
  2. const worker = new Worker("./worker.js", { type: "module", deno: true });
  3. worker.postMessage({ filename: "./log.txt" });
  4. // worker.js
  5. self.onmessage = async (e) => {
  6. const { filename } = e.data;
  7. const text = await Deno.readTextFile(filename);
  8. console.log(text);
  9. self.close();
  10. };
  11. // log.txt
  12. hello world
  1. $ deno run --allow-read --unstable main.js
  2. hello world

Deno 命名空间在worker中能使用时,worker继承了父进程的权限(即使用 --allow-* 标识的进程)。

官方也有意将workers权限做成可配置的。