【Canvas 系列】通过上下分层优化 Canvas 书写性能

前言

上一节 【Canvas系列】通过贝塞尔曲线解决 Canvas 书写的圆滑问题,我们通过点稀疏的方式,优化了 Canvas 的绘制性能。但在书写的过程中,因为点数量减少导致书写的效果不够理想,因此通过贝塞尔曲线的方式,将点连接起来,形成平滑的曲线,从而达到书写的效果。 这节,我们将通过上下分层的方式,优化 Canvas 的书写性能。

基本思路

这里我将 Canvas 分为上下两层,上层称之为动态层,下层称之为静态层,动态层用于书写,静态层用于显示书写的效果。当书写的时候,将书写的点绘制到上层,当书写完成后,将上层的内容绘制到下层,然后清空上层的内容,这样就可以达到书写的效果。

因为 Canvas 每次渲染都会将整个 Canvas 清空,所以我们需要将静态层的内容保存下来,然后在每次书写的时候,只需要渲染当前动态层的内容即可。

实现

创建 2 个 Canvas

首先创建 2 个 Canvas, 上层 Canvas 用于书写, 书写完成后将上层 Canvas 的内容绘制到下层 Canvas 中, 然后清空上层 Canvas 的内容, 这样就可以达到书写的效果。

这样的好处在于, 每次书写的时候, 只需要渲染上层 Canvas 的书写内容即可, 不需要每次都渲染整个 Canvas, 从而达到优化性能的目的。

html 复制代码
<style>
    #draw {
        border: 1px solid black;
        position: absolute;
        z-index: 9999;
    }

    #draw-content {
        border: 1px solid black;
        position: absolute;
        z-index: 9998;
        pointer-events: none;
    }
</style>

<!-- 动态层 Canvas -->
<canvas id="draw"></canvas>
<!-- 静态层 Canvas -->
<canvas id="draw-content"></canvas>

数据动静切换

笔者创建了 2 个 Canvas 后, 可以在书写的过程中(监听 pointerMove 事件),现在数据渲染到动态层 Canvas 上

当书写完成后(监听 pointerUp 事件),将动态层 Canvas 的内容渲染到静态层 Canvas 上

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

    canvas.addEventListener('pointermove', (e) => {
        if (!start) return; // 如果没有按下,则不绘制
        addPoint((e)); // 将鼠标移动的点添加到points数组中
        renderUpperCanvas(ctx); // 绘制上层
    });

    canvas.addEventListener('pointerup', (e) => {
        start = false;
        // 将上层 canvas 绘制的内容保存到下层 canvas 中
        history.push(points);
        points = []; // 绘制完毕后,清空points数组
        renderLowerCanvas(ctx, ctxContent);
    });
</script>

演示效果

这里我将动静 Canvas 分成了左右 2 个部分,左边是动态层 Canvas,右边是静态层 Canvas,可以看到,当书写的时候,只有左边的 Canvas 会有书写的效果,右边的 Canvas 不会有书写的效果,当书写完成后,左边的 Canvas 会将书写的内容绘制到右边的 Canvas 中,然后清空左边的 Canvas 的内容,这样就可以达到书写的效果。

当我们把 2 个 Canvas 整合在一起后,具体的效果如下

具体代码

传送门

相关推荐
牛奶13 分钟前
5MB vs 4KB vs 无限大:浏览器存储谁更强?
前端·浏览器·indexeddb
牛奶15 分钟前
setTimeout设为0就马上执行?JS异步背后的秘密
前端·性能优化·promise
Ailrid15 分钟前
@virid/core:用游戏引擎的思维来写应用-高度确定性的应用开发引擎
javascript
LaughingZhu2 小时前
Product Hunt 每日热榜 | 2026-04-05
前端·数据库·人工智能·经验分享·神经网络
SuperEugene2 小时前
Vue3 组件复用设计:Props / 插槽 / 组合式函数,三种复用方式选型|组件化设计基础篇
前端·javascript·vue.js
nFBD29OFC3 小时前
利用Vue元素指令自动合并tailwind类名
前端·javascript·vue.js
ISkp3V8b43 小时前
ASP.NET MVC]Contact Manager开发之旅之迭代2 - 修改样式,美化应用
前端·chrome
Highcharts.js4 小时前
高级可视化图表的暗色模式与主题|Highcharts 自适应主题配色全解
前端·react.js·实时图表
zk_one5 小时前
【无标题】
开发语言·前端·javascript
AIBox3656 小时前
openclaw api 配置排查与接入指南:网关启动、配置文件和模型接入全流程
javascript·人工智能·gpt