canvas使用贝塞尔曲线平滑拟合折线段的方法详解

本文主要介绍了基于canvas使用贝塞尔曲线平滑拟合折线段的方法的相关资料,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧,希望能帮助到大家。

写在最前

本次分享一下在canvas中将绘制出来的折线段的棱角“磨平”,也就是通过贝塞尔曲线穿过各个描点来代替原有的折线图。

为什么要平滑拟合折线段

先来看下Echarts下折线图的渲染效果:

canvas使用贝塞尔曲线平滑拟合折线段的方法详解 

一开始我没注意到其实这个折线段是曲线穿过去的,只认为是单纯的描点绘图,所以起初我实现的“简(丑)易(陋)”版本是这样的:

canvas使用贝塞尔曲线平滑拟合折线段的方法详解

不要关注样式,重点就是实现之后才发现看起来人家Echarts的实现描点非常的圆滑,也由此引发了之后的探讨。怎么有规律的画平滑曲线?

效果图

先来看下最终模仿的实现:

因为我也不知道Echarts内部怎么实现的(逃

canvas使用贝塞尔曲线平滑拟合折线段的方法详解 

canvas使用贝塞尔曲线平滑拟合折线段的方法详解 

看起来已经非常圆润了,和我们最初的设想十分接近了。再看下曲线是否穿过了描点:

canvas使用贝塞尔曲线平滑拟合折线段的方法详解 

好的!结果很明显现在来重新看下我们的实现方式。

实现过程

绘制折线图

贝塞尔曲线平滑拟合

模拟数据

var data = [Math.random() * 300];        for (var i = 1; i < 50; i++) { //按照echarts            data.push(Math.round((Math.random() - 0.5) * 20 + data[i - 1]));        }        option = {            canvas:{                id: 'canvas'            },            series: {                name: '模拟数据',                itemStyle: {                    color: 'rgb(255, 70, 131)'                },                areaStyle: {                    color: 'rgb(255, 158, 68)'                },                data: data            }        };

绘制折线图

首先初始化一个构造函数来放置需要用到的数据:

function LinearGradient(option) {    this.canvas = document.getElementById(option.canvas.id)    this.ctx = this.canvas.getContext('2d')    this.width = this.canvas.width    this.height = this.canvas.height    this.tooltip = option.tooltip    this.title = option.text    this.series = option.series //存放模拟数据}

绘制折线图:

LinearGradient.prototype.draw1 = function() { //折线参考线    ...     //要考虑到canvas中的原点是左上角,    //所以下面要做一些换算,    //diff为x,y轴被数据最大值和最小值的取值范围所平分的等份。    this.series.data.forEach(function(item, index) {        var x = diffX * index,            y = Math.floor(self.height - diffY * (item - dataMin))        self.ctx.lineTo(x, y) //绘制各个数据点    })    ...}

贝塞尔曲线平滑拟合

贝塞尔曲线的关键点在于控制点的选择,这个网站可以动态的展现控制点不同而绘制的不同的曲线。而对于控制点的计算。。作者还是选择了百度一下毕竟数学不好:)。具体算法有兴趣的同学可以深入了解下,现在直接说下计算控制点的结论。

canvas使用贝塞尔曲线平滑拟合折线段的方法详解

上面的公式涉及到四个坐标点,当前点,前一个点以及后两个点,而当坐标值为下图展示的时候绘制出来的曲线如下所示:

canvas使用贝塞尔曲线平滑拟合折线段的方法详解

不过会有一个问题就是起始点和最后一个点不能用这个公式,不过那篇文章也给出了边界值的处理办法:

canvas使用贝塞尔曲线平滑拟合折线段的方法详解 

所以在将折线换成平滑曲线的时候,将边界值以及其他控制点计算好之后代入到贝塞尔函数中就完成了:

//核心实现this.series.data.forEach(function(item, index) { //找到前一个点到下一个点中间的控制点    var scale = 0.1 //分别对于ab控制点的一个正数,可以分别自行调整    var last1X = diffX * (index - 1),        last1Y = Math.floor(self.height - diffY * (self.series.data[index - 1] - dataMin)),        //前一个点坐标        last2X = diffX * (index - 2),        last2Y = Math.floor(self.height - diffY * (self.series.data[index - 2] - dataMin)),        //前两个点坐标        nowX = diffX * (index),        nowY = Math.floor(self.height - diffY * (self.series.data[index] - dataMin)),        //当期点坐标        nextX = diffX * (index + 1),        nextY = Math.floor(self.height - diffY * (self.series.data[index + 1] - dataMin)),        //下一个点坐标        cAx = last1X + (nowX - last2X) * scale,        cAy = last1Y + (nowY - last2Y) * scale,        cBx = nowX - (nextX - last1X) * scale,        cBy = nowY - (nextY - last1Y) * scale     if(index === 0) {        self.ctx.lineTo(nowX, nowY)        return    } else if(index ===1) {        cAx = last1X + (nowX - 0) * scale        cAy = last1Y + (nowY - self.height) * scale     } else if(index === self.series.data.length - 1) {        cBx = nowX - (nowX - last1X) * scale        cBy = nowY - (nowY - last1Y) * scale    }         self.ctx.bezierCurveTo(cAx, cAy, cBx, cBy, nowX, nowY);        //绘制出上一个点到当前点的贝塞尔曲线    })

由于我每次遍历的点都是当前点,但是文章中给出的公式是计算会知道下一个点的控制点算法,故在代码实现中我将所有点的计算挪前了一位。当index = 0时也就是初始点是不需要曲线绘制的,因为我们绘制的是从前一个点到当前点的曲线,没有到0的曲线需要绘制。从index = 1开始我们就可以正常开始绘制,从0到1的曲线,由于index = 1时是没有在他前面第二个点的故其属于边界值点,也就是需要特殊进行计算,以及最后一个点。其余均按照正常公式算出AB的xy坐标代入贝塞尔函数即可。

相关推荐:

canvas实现高阶贝塞尔曲线

使用CSS做贝塞尔曲线

贝塞尔曲线的应用详解

以上就是canvas使用贝塞尔曲线平滑拟合折线段的方法详解的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1545506.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月21日 17:01:27
下一篇 2025年12月21日 17:01:48

相关推荐

  • 如何用CSS实现平滑滚动效果

    如何用CSS实现平滑滚动效果 在网页设计与开发中,滚动效果是一种非常常见且炫酷的效果,能够为用户带来更好的体验。而实现平滑滚动效果,可以通过CSS的一些技巧来实现。本文将介绍如何使用CSS来实现平滑滚动效果,并提供具体的代码示例。 一、使用锚点实现内部页面平滑滚动 锚点是HTML中的一个标记,可以将…

    2025年12月24日
    000
  • 如何通过纯CSS实现网页的平滑滚动背景效果

    如何通过纯CSS实现网页的平滑滚动背景效果 背景是网页设计中非常重要的一部分,可以增强页面的视觉效果和用户体验。传统的网页背景通常是静态的,但是通过使用纯CSS技术,我们可以实现一种平滑滚动背景效果,从而为网页添加更加动感和生动的视觉效果。在本文中,我们将介绍如何使用CSS来实现平滑滚动背景效果,并…

    2025年12月24日
    000
  • css3实现把图片画到画布上(代码实例)

    本文给大家介绍把图片画到画布上的方法,适应pc和移动端 使用。有一定的参考价值,有需要的朋友可以参考一下,希望对你们有所帮助。 画一张图片到画布上 您的浏览器不支持canvas标签。 var canvas = document.getElementById(“myCanvas”);//获取画笔var…

    2025年12月24日
    000
  • canvas线条的属性解析

    本文主要和大家介绍了canvas线条的属性详解的相关资料,小编觉得挺不错的,现在分享给大家,希望能帮助到大家。 一、线条的帽子lineCap 取值:butt(默认值),round圆头,square方头 var canvas=document.getElementById(“canvas”);canv…

    2025年12月24日 好文分享
    000
  • html5怎么创建图片文件_HTML5 Canvas图片生成方法

    HTML5通过Canvas API可生成图片文件。首先创建canvas元素并设置宽高,接着用JavaScript获取2D上下文绘制图形或加载图片,然后调用toDataURL()或toBlob()将画布内容转为图片数据,最后实现下载。示例中绘制了背景、矩形和文字,并通过按钮触发saveImage()函…

    2025年12月23日
    100
  • canvas标签是用来做什么的

    canvas用于通过JavaScript在网页上绘制图形,提供像素级控制,适合高性能动态图形场景。 canvas 标签主要用来在网页上通过JavaScript绘制图形。它提供了一块空白的绘图区域,你可以用代码在这上面画出各种形状、文字、图片,甚至复杂的动画和游戏界面。它本质上是一个位图(bitmap…

    2025年12月22日
    000
  • canvas如何绘制圆形

    答案:Canvas绘制圆形使用arc()方法,通过设置圆心、半径、起始角度和结束角度可绘制实心圆、空心圆、扇形和圆环,并可通过减少状态切换、使用离屏Canvas等方法优化性能。 绘制圆形,Canvas API 提供了 arc() 方法,掌握它就能轻松搞定。不过,圆形虽简单,但要画得漂亮、用得灵活,还…

    2025年12月22日
    000
  • svg和canvas有什么区别

    SVG适合静态图形与交互,Canvas适合高性能动态渲染。SVG基于矢量,可无损缩放,支持DOM操作、动画及可访问性,适用于图标、响应式设计;但复杂图形性能差。Canvas基于像素,直接操作画布,渲染效率高,适合游戏、数据可视化,但缩放失真、缺乏可访问性。选择需综合图形类型、性能、交互、可维护性等需…

    2025年12月22日
    000
  • canvas如何绘制矩形

    答案:在HTML5 Canvas上绘制矩形需获取2D上下文,使用fillRect()填充、strokeRect()描边、clearRect()清除;通过fillStyle、strokeStyle、lineWidth等属性控制颜色与边框,结合路径方法可绘制圆角矩形,并需注意Canvas坐标系及实际尺寸…

    2025年12月22日
    000
  • canvas如何保存绘图状态

    Canvas通过context.save()和context.restore()管理绘图状态,前者保存当前样式、变换、剪辑路径等状态到栈中,后者恢复最近保存的状态,确保局部操作不影响全局绘制。 Canvas保存绘图状态主要依靠 context.save() 和 context.restore() 这…

    2025年12月22日
    000
  • HTML如何制作画板?canvas绘图怎么实现?

    canvas绘图的核心概念包括:1. 路径(paths),通过beginpath()开始新路径,moveto()移动起点,lineto()画线,arc()画弧,rect()画矩形,再用stroke()描边或fill()填充;2. 样式(styles),strokestyle设置线条颜色,fillst…

    2025年12月22日
    000
  • 什么是canvas?HTML5画布如何使用?

    canvas的绘图能力依赖javascript实现,其基本使用步骤为:1. 在html中添加元素并设置id、width和height属性;2. 使用javascript通过document.getelementbyid()获取canvas元素,并调用getcontext(‘2d&#821…

    2025年12月22日
    000
  • HTML如何实现手写签名?canvas怎么捕捉笔画?

    html实现手写签名的核心是利用canvas元素,通过javascript监听鼠标或触摸事件来捕捉笔画轨迹并绘制。1. 首先在html中创建一个canvas元素并设置id和尺寸;2. 使用css设置外观样式,如边框和固定大小;3. 在javascript中获取canvas及其2d绘图上下文,定义is…

    2025年12月22日
    000
  • canvas标签的用途是什么?绘图功能怎么实现?

    canvas标签本身是空白画布,无绘图能力,需通过javascript的getcontext(‘2d’)获取2d渲染上下文进行绘图;2. 绘图步骤包括定义canvas元素、获取上下文、设置样式与路径、调用fill或stroke等方法绘制图形;3. canvas基于像素,适合高…

    2025年12月22日 好文分享
    000
  • HTML5的Canvas元素能做什么?如何绘制基本图形?

    canvas和svg的主要区别在于canvas基于像素,适合大量图形处理和像素控制,而svg基于矢量,适合可缩放图形和交互。1. canvas是位图,缩放可能失真;2. svg是矢量图,缩放不失真;3. canvas需手动管理状态,无内置对象模型;4. svg有dom结构,便于交互;5. canva…

    2025年12月22日 好文分享
    000
  • HTML的canvas标签怎么绘制图形?有哪些基本API?

    canvas和svg的选择取决于具体需求。1.canvas基于像素,适合处理大量图形、游戏、动画等高性能场景,但不支持直接修改图形元素。2.svg基于矢量,适合图表、地图等需要缩放和频繁修改的场景,且具备更好的可访问性。3.canvas性能在复杂图形渲染上更强,而svg在少量动态图形时更高效。4.若…

    2025年12月22日 好文分享
    000
  • HTML怎么设置Canvas画布?

    要设置html canvas画布,1. 使用标签创建画布元素并设置id、宽高和样式;2. 通过javascript获取canvas对象和上下文;3. 使用canvas api进行绘制。canvas的width和height属性在html中定义像素尺寸,在javascript中修改会清空内容。优化性能…

    2025年12月22日 好文分享
    000
  • html中canvas标签作用 html中canvas绘制图形基础

    canvas 标签在 html 中主要用于通过 javascript 动态渲染图形、图像和其他视觉元素。1. 它本身是一个容器,不具备绘图能力,需依赖 javascript 提供的上下文进行绘制;2. 绘制图形的基本步骤包括获取 canvas 元素、获取 2d 渲染上下文、使用上下文方法绘制图形并呈…

    2025年12月22日 好文分享
    000
  • HTML如何用JS操作Canvas?绘图API与动画实现教程

    js通过canvas api操作canvas元素实现图形绘制与动画效果,首先获取上下文并调用api绘图,结合requestanimationframe创建动画。1.定义canvas元素并指定id和尺寸;2.使用js获取canvas元素及其2d渲染上下文,若失败则提示错误;3.使用ctx对象绘制矩形、…

    2025年12月22日 好文分享
    000
  • 学习如何在canvas上绘制图形

    如何使用canvas图形绘制 Canvas是HTML5中的一个功能强大的元素,它允许我们使用JavaScript绘制图形、动画、游戏等。在本篇文章中,我们将学习如何使用canvas元素绘制图形,并通过具体的代码示例来帮助我们更好地理解。 一、准备工作在开始之前,我们需要一个HTML文档,其中包含一个…

    2025年12月22日
    000

发表回复

登录后才能评论
关注微信