React项目使用 Web Worker

缘起

Web Worker 老生常谈了 ------ 它允许你在主线程之外创建额外的线程来执行任务,例如处理文件,埋点轮询,如何在 React 中使用呢 ?

webpack5^

JSX

typescript 复制代码
  useEffect(() => {
    const worker = new Worker(new URL('./wk.ts', import.meta.url));

    // set Worker Handle
    worker.onmessage = function (e: MessageEvent) {
      const result: string = e.data;
      console.log('Received result from Worker:', result);
    };

    // post Worker message
    worker.postMessage('hello from main thread');
  }, []);

wk.ts

typescript 复制代码
self.onmessage = function(e: MessageEvent) {
  const data: string = e.data;  
  //  This's Worker in data 
  const result: string = data.toUpperCase();
  
  // Post result to JSX
  self.postMessage(result);
};
  • import.meta 是一个内置在 ES 模块内部的对象,import.meta.url 表示一个模块在浏览器和 Node.js 的绝对路径。该特性属于 es2020 的一部分
  • new URL传入 path & base 写入内存

worker-loader

不是 webpack5^ 可以使用插件 Loader worker-loader

js 复制代码
module.exports = {
  module: {
    rules: [
      {
        // 以 .worker.js 结尾的文件将被 worker-loader 加载
        test: /\.worker\.(c|m)?js$/i,
        use: {
          loader: "worker-loader",
        },
      },
    ],
  },
};

or

js 复制代码
  chainWebpack(config) {
    config.module
      .rule('worker')
      .test(/\.worker\.ts$/)
      .use('worker-loader')
      .loader('worker-loader')
      .end();
  },

为了保证 worker 中的代码被 babel 转译,可以让 babel-loaderworker-loader 之前执行。ts-loader 同理

why import worker file ?

js 复制代码
import workPath from "./worker.js";
const worker = new Worker(workPath);

为什么不能直接这样导入worker 文件呢?

同样也是需要借助特定的 loader,类似于 file-loader。至于 worker-loader 则是将new Worker(workPath)的步骤内置到 loader 处理流程了,并导出一个函数,外面直接使用该函数即可创建指定的 Worker

js 复制代码
module.exports = function () {
  return new Worker(__webpack_public_path_ + "123abc.worker.js");
};

worker-loader 把文件转成类似👆

blobUrl

还有一种方式就是 ------ worker.js 的主函数转化为 blobUrl 导出,供主线程引用。该方法的好处是可以动态创建 worker

js 复制代码
// worker.js
const contentCode = function () {} // worker 脚本主函数
const blob = new Blob([contentCode.toString()], {type: 'text/javascript'});
export {url: URL.createObjectURL(blob)}


// main.js
import { url } from './worker.js'
const worker = new Worker(url);

Reference: worker-loader源码

相关推荐
用户990450177800919 小时前
ruoyi集成dmn规则引擎
前端
袋鱼不重19 小时前
AI入门知识点:什么是 AIGC、多模态、RAG、Function Call、Agent、MCP?
前端·aigc·ai编程
NuLL19 小时前
空值检测工具函数-统一规范且允许自定义配置的空值检测方案
前端
栀秋66619 小时前
“无重复字符的最长子串”:从O(n²)哈希优化到滑动窗口封神,再到DP降维打击!
前端·javascript·算法
xhxxx19 小时前
不用 Set,只用两个布尔值:如何用标志位将矩阵置零的空间复杂度压到 O(1)
javascript·算法·面试
鹿鹿鹿鹿isNotDefined19 小时前
Antd5.x 在 Next.js14.x 项目中,初次渲染样式丢失
前端·react.js·next.js
梨子同志19 小时前
Node.js 工具模块详解
前端
有意义19 小时前
斐波那契数列:从递归到优化的完整指南
javascript·算法·面试
谷歌开发者19 小时前
Web 开发指向标|AI 辅助功能在性能面板中的使用与功能
前端·人工智能
OpenTiny社区19 小时前
TinyEngine2.9版本发布:更智能,更灵活,更开放!
前端·vue.js·低代码