【Canvas 系列】通过 OffscreenCanvas + Worker 提高书写性能

前言

上一节我们通过【Canvas系列】通过离屏渲染提高 Canvas 书写性能,但是在绘制的过程中,我们会发现,当绘制的图形越来越多时,Canvas 的渲染性能会越来越差,这是因为我们在绘制图形会阻塞主线程,如果主线程中还有其他的任务也会表现出卡顿的效果,Canvas 的渲染性能越来越差。

这节我们将通过 OffscreenCanvas + Worker 将绘制图形的任务放到 Worker 中进行,避免阻塞主线程,从而提高 Canvas 的渲染性能。

实现思路

在 worker 线程中是无法操作 DOM 的,但 OffscreenCanvas 可以在 worker 线程中进行操作,因此我们可以通过 OffscreenCanvas 将绘制图形的任务放到 worker 线程中进行。这样可以减少主线程的任务,从而提高书写性能。

具体实现

创建 OffscreenCanvas

js 复制代码
const canvas = document.getElementById('draw');
const offScreenCanvas = canvas.transferControlToOffscreen(); // 将 canvas 转换为 offScreenCanvas

将 offScreenCanvas 传递给 worker 线程

js 复制代码
const worker = new Worker('./worker.js'); // 创建一个 webWorker
const offScreenCanvas = canvas.transferControlToOffscreen(); // 将 canvas 转换为 offScreenCanvas

worker.postMessage({
    type: 'init',
    offScreenCanvas,
    width: width,
    height: height,
    dpr,
}, [offScreenCanvas]); // 将 offScreenCanvas 传递给 webWorker


worker.onmessage = function (e) {
    const type = e.data.type;
    switch (type) {
        case 'init':
            init = true; // 判断是否初始化完毕
            break;
    }
}

通过监听事件获取书写的坐标点传递给 worker 线程, 然后在 worker 线程中进行绘制

js 复制代码
/**
 * 自由画笔的实现思路
 * 1 监听鼠标事件
 * 2 将鼠标移动的轨迹记录下来
 * 3 然后将这些点连接成线
 */
canvas.addEventListener("pointerdown", (e) => {
    if (!init) return;
    start = true; // 通过监听鼠标按下事件,来判断是否开始绘制
    addPoint(e); // 将鼠标按下的点添加到points数组中
});

canvas.addEventListener(
    "pointermove",
    (e) => {
        console.log("e--->throttle", e);
        if (!start) return; // 如果没有按下,则不绘制
        addPoint(e); // 将鼠标移动的点添加到points数组中
        init && worker.postMessage({
            type: 'render',
            points,
        }); // 将 点 传递给 webWorker
    }
);

canvas.addEventListener("pointerup", (e) => {
    start = false;
    points = []; // 绘制完毕后,清空points数组
});


/*
* 将鼠标事件的点转化为相对于canvas的坐标上的点
*/
function addPoint(e) {
    const x = e.clientX;
    const y = e.clientY;
    points.push({
        x,
        y,
    });
}

实现效果

具体代码

参考文章

相关推荐
知识分享小能手2 小时前
Vue3 学习教程,从入门到精通,Axios 在 Vue 3 中的使用指南(37)
前端·javascript·vue.js·学习·typescript·vue·vue3
程序员码歌5 小时前
【零代码AI编程实战】AI灯塔导航-总结篇
android·前端·后端
用户21411832636026 小时前
免费玩转 AI 编程!Claude Code Router + Qwen3-Code 实战教程
前端
小小愿望7 小时前
前端无法获取响应头(如 Content-Disposition)的原因与解决方案
前端·后端
小小愿望7 小时前
项目启功需要添加SKIP_PREFLIGHT_CHECK=true该怎么办?
前端
烛阴7 小时前
精简之道:TypeScript 参数属性 (Parameter Properties) 详解
前端·javascript·typescript
海上彼尚8 小时前
使用 npm-run-all2 简化你的 npm 脚本工作流
前端·npm·node.js
开发者小天8 小时前
为什么 /deep/ 现在不推荐使用?
前端·javascript·node.js
如白驹过隙9 小时前
cloudflare缓存配置
前端·缓存
excel9 小时前
JavaScript 异步编程全解析:Promise、Async/Await 与进阶技巧
前端