【数据可视化】数据可视化Canvas

news2025/1/13 7:37:01

在这里插入图片描述

1、了解Canvas

◼什么是Canvas
---- Canvas 最初由Apple于2004 年引入,用于Mac OS X WebKit组件,为仪表板小部件和Safari浏览器等应用程序提供支持。后来,它被Gecko内核的浏览器(尤其是Mozilla Firefox),Opera和Chrome实现,并被网页超文本应用技术工作小组提议为下一代的网络技术的标准元素(HTML5新增元素)。
---- Canvas提供了非常多的JavaScript绘图API(比如:绘制路径、矩形、圆、文本和图像等方法),与元素可以绘制各种2D图形。
---- Canvas API 主要聚焦于2D 图形。当然也可以使用元素对象的WebGL API 来绘制2D 和3D 图形。
◼Canvas的应用场景
---- 可以用于动画、游戏画面、数据可视化、图片编辑以及实时视频处理等方面。
◼Canvas 浏览器兼容性

在这里插入图片描述

2、Canvas 优缺点

◼Canvas 优点:
---- Canvas提供的功能更原始,适合像素处理,动态渲染和数据量大的绘制,如:图片编辑、热力图、炫光尾迹特效等。
---- Canvas非常适合图像密集型的游戏开发,适合频繁重绘许多的对象。
---- Canvas能够以.png 或.jpg 格式保存结果图像,适合对图片进行像素级的处理。
◼Canvas 缺点:
---- 在移动端可以能会因为Canvas数量多,而导致内存占用超出了手机的承受能力,导致浏览器崩溃。
---- Canvas 绘图只能通过JavaScript脚本操作(allinjs)。
---- Canvas 是由一个个像素点构成的图形,放大会使图形变得颗粒状和像素化,导致模糊。

3、初体验Canvas

◼ 使用Canvas的注意事项:
---- 和 元素很相像,唯一的不同就是它并没有src 和alt 属性。
---- 标签只有两个属性——width和height(单位默认为px)。当没有设置宽度和高度时,canvas 会初始化宽为300px 和高为150px。
----与 元素不同, 元素必须需要结束标签()。如结束标签不存在,则文档其余部分会被认为是替代内容,将不会显示出来。
----测试canvas.getContext() 方法的存在,可以检查浏览器是否支持Canvas。
◼初体验Canvas
---- Canvas通用模板
---- Canvas绘制正方形
在这里插入图片描述

4、Canvas Grid和坐标空间

◼在开始画图之前,我们需要了解一下Canvas网格(canvas grid)和坐标系。
◼Canvas Grid 或坐标空间
----假如,HTML 模板中有个宽150px, 高150px 的元素。元素默认被网格所覆盖。
----通常来说网格中的一个单元相当于canvas 元素中的一像素。
----该网格的原点位于坐标(0,0) 的左上角。所有元素都相对于该原点放置。
----网格也可以理解为坐标空间(坐标系),坐标原点位于canvas元素的左上角,被称为初始坐标系;如右图中蓝色正方形,左上角的坐标为距离左边x 像素,距离上边y像素,坐标为(x,y)
----网格或坐标空间是可以变换的,后面会讲如何将原点转换到不同的位置,旋转网格甚至缩放它。
注意:移动了原点后,默认所有后续变换都将基于新坐标系的变换。
在这里插入图片描述

5、绘制矩形(Rectangle)

◼Canvas支持两种方式来绘制矩形:矩形方法和路径方法。
---- 路径是通过不同颜色和宽度的线段或曲线相连形成的不同形状的点的集合。
---- 除了矩形,其他的图形都是通过一条或者多条路径组合而成的。
---- 通常我们会通过众多的路径来绘制复杂的图形。
◼Canvas 绘图的矩形方法:
---- fillRect(x, y, width, height):绘制一个填充的矩形
---- strokeRect(x, y, width, height):绘制一个矩形的边框
---- clearRect(x, y, width, height):清除指定矩形区域,让清除部分完全透明。
◼方法参数:
---- 上面的方法都包含了相同的参数。
---- x 与y 指定了在canvas画布上所绘制矩形的左上角(相对于原点)的坐标(不支持undefined)。
---- width 和height 设置矩形的尺寸。
在这里插入图片描述

6、认识路径

◼ 什么是路径?
---- 图形的基本元素是路径。路径是通过不同颜色和宽度的线段或曲线相连形成的不同形状的点的集合。
---- 路径是可由很多子路径构成,这些子路径都是在一个列表中,列表中所有子路径(线、弧形等)将构成图形。
---- 一个路径,甚至一个子路径,通常都是闭合的。
◼ 使用路径绘制图形的步骤:
---- 1.首先需要创建路径起始点(beginPath)。
---- 2.然后使用画图命令去画出路径(arc、lineTo)。
---- 3.之后把路径闭合(closePath,不是必须)。
---- 4.一旦路径生成,就能通过描边(stroke)或填充路径区域(fill)来渲染图形。
◼ 以下是绘制路径时,所要用到的函数
---- beginPath():新建一条路径,生成之后,图形绘制命令被指向到新的路径上绘图,不会关联到旧的路径。
---- closePath():闭合路径之后图形绘制命令又重新指向到beginPath之前的上下文中。
---- stroke():通过线条来绘制图形轮廓/描边(针对当前路径图形)。
---- fill():通过填充路径的内容区域生成实心的图形(针对当前路径图形)。
在这里插入图片描述
在这里插入图片描述

7、路径-绘制直线

◼ 移动画笔(moveTo)方法
---- moveTo方法是不能画出任何东西,但是它也是路径列表的一部分
---- moveTo可以想象为在纸上作业,一支钢笔或者铅笔的笔尖从一个点到另一个点的移动过程。
---- moveTo(x, y):将笔移动到指定的坐标x 、y 上。
---- 当canvas 初始化或者beginPath()调用后,我们通常会使用moveTo(x,y)函数设置起点。
---- 使用moveTo函数能够绘制一些不连续的路径。
在这里插入图片描述
◼ 绘制直线(lineTo)方法
---- lineTo(x, y):绘制一条从当前位置到指定(x ,y)位置的直线。
---- 该方法有两个参数(x ,y)代表坐标系中直线结束的点。
---- 开始点和之前的绘制路径有关,之前路径的结束点就是接下来的开始点。
---- 当然开始点也可以通过moveTo(x,y)函数改变。
◼ 绘制一条直线
---- 第一步:调用beginPath() 来生成路径。本质上,路径是由很多子路径(线、弧形、等)构成。
---- 第二步:调用moveTo、lineTo函数来绘制路径(路径可以是连续也可以不连续)。
---- 第三步:闭合路径closePath(),虽然不是必需的,但是通常都是要闭合路径。
---- 第四步:调用stroke()函数来给直线描边。
在这里插入图片描述

8、路径-绘制三角形(Triangle)

◼ 绘制一个三角形步骤
---- 第一步:调用beginPath() 来生成路径。
---- 第二步:调用moveTo()、lineTo()函数来绘制路径。
---- 第三步:闭合路径closePath(),不是必需的。
1)closePath()方法会通过绘制一条从当前点到开始点的直线来闭合图形。
2)如果图形是已经闭合了的,即当前点为开始点,该函数什么也不做。
---- 第四步:调用stroke()函数来给线描边,或者调用fill()函数来填充(使用填充fill 时,路径会自动闭合,而stroke不会)。
在这里插入图片描述

9、路径-绘制圆弧(Arc)、圆(Circle)

◼ 绘制圆弧或者圆,使用arc()方法。
---- arc(x, y, radius, startAngle, endAngle, anticlockwise),该方法有六个参数:
1) x、y:为绘制圆弧所在圆上的圆心坐标。
2) radius:为圆弧半径。
3) startAngle、endAngle:该参数用弧度定义了开始以及结束的弧度。这些都是以x 轴为基准。
4) anticlockwise:为一个布尔值。为true ,是逆时针方向,为false,是顺时针方向,默认为false。
◼ 计算弧度
---- arc() 函数中表示角的单位是弧度,不是角度。
---- 角度与弧度的JS表达式:弧度=( Math.PI / 180 ) * 角度,即1角度=Math.PI/180个弧度
1) 比如:旋转90°:Math.PI / 2;旋转180°:Math.PI ;旋转360°:Math.PI * 2;旋转-90°:-Math.PI / 2;
◼ 绘制一个圆弧的步骤
---- 第一步:调用beginPath() 来生成路径。
---- 第二步:调用arc()函数来绘制圆弧。
---- 第三步:闭合路径closePath(),不是必需的。
---- 第四步:调用stroke()函数来描边,或者调用fill()函数来填充(使用填充fill 时,路径会自动闭合)
在这里插入图片描述

10、路径-矩形(Rectangle)

◼ 绘制矩形的另一个方法:
---- 调用rect() 函数绘制,即将一个矩形路径增加到当前路径上
---- rect(x, y, width, height)
---- 绘制一个左上角坐标为(x,y),宽高为width 以及height 的矩形。
◼ 注意:
---- 当该方法执行的时候,moveTo(x,y) 方法自动设置坐标参数(0,0)。也就是说,当前笔触自动重置回默认坐标。
在这里插入图片描述

11、路径-绘制圆弧(Arc)、圆(Circle)

◼ 绘制圆弧或者圆,使用arc()方法。
---- arc(x, y, radius, startAngle, endAngle, anticlockwise),该方法有六个参数:
1) x、y:为绘制圆弧所在圆上的圆心坐标。
2) radius:为圆弧半径。
3) startAngle、endAngle:该参数用弧度定义了开始以及结束的弧度。这些都是以x 轴为基准。
4) anticlockwise:为一个布尔值。为true ,是逆时针方向,为false,是顺时针方向,默认为false。
◼ 计算弧度
---- arc() 函数中表示角的单位是弧度,不是角度。
---- 角度与弧度的JS表达式:弧度=( Math.PI / 180 ) * 角度,即1角度=Math.PI/180个弧度
---- 比如:旋转90°:Math.PI / 2;旋转180°:Math.PI ;旋转360°:Math.PI * 2;旋转-90°:-Math.PI / 2;
◼ 绘制一个圆弧的步骤
---- 第一步:调用beginPath() 来生成路径。
---- 第二步:调用arc()函数来绘制圆弧。
---- 第三步:闭合路径closePath(),不是必需的。
---- 第四步:调用stroke()函数来描边,或者调用fill()函数来填充(使用填充fill 时,路径会自动闭合)。
在这里插入图片描述

12、路径-矩形(Rectangle)

◼ 绘制矩形的另一个方法:
---- 调用rect() 函数绘制,即将一个矩形路径增加到当前路径上
---- rect(x, y, width, height)
---- 绘制一个左上角坐标为(x,y),宽高为width 以及height 的矩形。
◼ 注意:
---- 当该方法执行的时候,moveTo(x,y) 方法自动设置坐标参数(0,0)。也就是说,当前笔触自动重置回默认坐标。
在这里插入图片描述

13、色彩Colors

◼ 前面已经学过了很多绘制图形的方法。如果我们想要给图形上色,有两个重要的属性可以做到:
---- fillStyle = color:设置图形的填充颜色,需在fill()函数前调用。
---- strokeStyle = color:设置图形轮廓的颜色,需在stroke()函数前调用。
在这里插入图片描述

◼ color颜色
---- color 可以是表示CSS 颜色值的字符串,支持:关键字、十六进制、rgb、rgba格式。
---- 默认情况下,线条和填充颜色都是黑色(CSS 颜色值#000000)。
◼ 注意
---- 一旦设置了strokeStyle 或者fillStyle 的值,那么这个新值就会成为新绘制的图形的默认值。
---- 如果你要给图形上不同的颜色,你需要重新设置fillStyle 或strokeStyle 的值。
◼ 额外补充
---- fill() 函数是图形填充,fillStyle属性是设置填充色
---- stroke() 函数是图形描边,strokeStyle属性是设置描边色
在这里插入图片描述

14、透明度Transparent

◼ 除了可以绘制实色图形,我们还可以用canvas 来绘制半透明的图形。
◼ 方式一:strokeStyle 和fillStyle属性结合RGBA:
在这里插入图片描述

◼ 方式二:globalAlpha 属性
---- globalAlpha = 0~1
✓ 这个属性影响到canvas 里所有图形的透明度
✓ 有效的值范围是0.0(完全透明)到1.0(完全不透明),默认是1.0。
在这里插入图片描述

15、线型Line styles

调用lineTo()函数绘制的线条,是可以通过一系列属性来设置线的样式
---- lineWidth = value:设置线条宽度。
---- lineCap = type:设置线条末端样式。
---- lineJoin = type:设定线条与线条间接合处的样式。
---- …
◼ lineWidth
---- 设置线条宽度的属性值必须为正数。默认值是1.0px,不需单位。(零、负数、Infinity和NaN值将被忽略)
---- 线宽是指给定路径的中心到两边的粗细。换句话说就是在路径的两边各绘制线宽的一半。
---- 如果你想要绘制一条从(3,1) 到(3,5),宽度是1.0 的线条,你会得到像第二幅图一样的结果。
---- 路径的两边个各延伸半个像素填充并渲染出1像素的线条(深蓝色部分)
---- 两边剩下的半个像素又会以实际画笔颜色一半色调来填充(浅蓝部分)
---- 实际画出线条的区域为(浅蓝和深蓝的部分),填充色大于1像素了,这就是为何宽度为1.0 的线经常并不准确的原因。
---- 要解决这个问题,必须对路径精确的控制。如,1px的线条会在路径两边各延伸半像素,那么像第三幅图那样绘制从(3.5 ,1) 到(3.5, 5) 的线条,其边缘正好落在像素边界,填充出来就是准确的宽为1.0 的线条。
在这里插入图片描述
◼ lineCap:属性的值决定了线段端点显示的样子。它可以为下面的三种的其中之一:
---- butt 截断,默认是butt。
---- round 圆形
---- square 正方形
◼ lineJoin:属性的值决定了图形中线段连接处所显示的样子。它可以是这三种之一:
---- round 圆形
---- bevel 斜角
---- miter 斜槽规,默认是miter。
在这里插入图片描述

16、绘制文本

◼ canvas 提供了两种方法来渲染文本:
---- fillText(text, x, y [, maxWidth])
---- 在(x,y) 位置,填充指定的文本
---- 绘制的最大宽度(可选)。
---- strokeText(text, x, y [, maxWidth])
---- 在(x,y) 位置,绘制文本边框
---- 绘制的最大宽度(可选)。
◼ 文本的样式(需在绘制文本前调用)
---- font = value:当前绘制文本的样式。这个字符串使用和CSS font 属性相同的语法。默认的字体是:10px sans-serif。
---- textAlign = value:文本对齐选项。可选的值包括:start, end, left, right or center. 默认值是start
---- textBaseline = value:基线对齐选项。可选的值包括:top, hanging, middle, alphabetic, ideographic, bottom。
---- 默认值是alphabetic。
在这里插入图片描述

17、绘制图片

◼ 绘制图片,可以使用drawImage 方法将它渲染到canvas 里。drawImage 方法有三种形态:
---- drawImage(image, x, y)
---- 其中image 是image 或者canvas 对象,x 和y 是其在目标canvas 里的起始坐标。
---- drawImage(image, x, y, width, height)
---- 这个方法多了2 个参数:width 和height,这两个参数用来控制当向canvas 画入时应该缩放的大小
---- drawImage(image, sx, sy, sWidth, sHeight, dx, dy, dWidth, dHeight)
---- 第一个参数和其它的是相同的,都是一个图像或者另一个canvas 的引用。其它8 个参数最好是参照右边的图解,前4 个是定义图像源的切片位置和大小,后4 个则是定义切片的目标显示位置和大小。
◼ 图片的来源,canvas 的API 可以使用下面这些类型中的一种作为图片的源:
---- HTMLImageElement:这些图片是由Image()函数构造出来的,或者任何的元素。
---- HTMLVideoElement:用一个HTML 的元素作为你的图片源,可以从视频中抓取当前帧作为一个图像。
---- HTMLCanvasElement:可以使用另一个 元素作为你的图片源。
---- …

18、Canvas绘画状态-保存和恢复

◼ Canvas绘画状态
---- 是当前绘画时所产生的样式和变形的一个快照。
---- Canvas在绘画时,会产生相应的绘画状态,其实我们是可以将某些绘画的状态存储在栈中来为以后复用。
---- Canvas 绘画状态的可以调用save 和restore 方法是用来保存和恢复,这两个方法都没有参数,并且它们是成对存在的。
◼ 保存和恢复(Canvas)绘画状态
---- save():保存画布(canvas) 的所有绘画状态
---- restore():恢复画布(canvas) 的所有绘画状态
◼ Canvas绘画状态包括:
---- 当前应用的变形(即移动,旋转和缩放)
---- 以及这些属性:strokeStyle, fillStyle, globalAlpha, lineWidth, lineCap, lineJoin, miterLimit,
shadowOffsetX, shadowOffsetY, shadowBlur, shadowColor, font, textAlign, textBaseline…
---- 当前的裁切路径(clipping path)
在这里插入图片描述

19、变形Transform

◼ Canvas和CSS3一样也是支持变形,形变是一种更强大的方法,可以将坐标原点移动到另一点、形变可以对网格进行旋转和缩放。
◼ Canvas的形变有4种方法实现:
---- translate(x, y):用来移动canvas 和它的原点到一个不同的位置。
---- x 是左右偏移量,y是上下偏移量(无需要单位),如右图所示。
---- rotate(angle):用于以原点为中心旋转canvas,即沿着z轴旋转。
---- angle是旋转的弧度,是顺时针方向,以弧度为单位。
---- scale(x, y):用来增减图形在canvas 中像素数目,对图形进行缩小或放大。
---- x 为水平缩放因子,y 为垂直缩放因子。如果比1 小,会缩小图形,如果比1 大会放大图形。默认值为1,也支持负数。
---- transform(a, b, c, d, e, f):允许对变形矩阵直接修改。这个方法是将当前的变形矩阵乘上一个基于自身参数的矩阵。
◼ 注意事项:
---- 在做变形之前先调用save方法保存状态是一个良好的习惯。
---- 大多数情况下,调用restore 方法比手动恢复原先的状态要简单得多。
---- 如果在一个循环中做位移但没有保存和恢复canvas状态,很可能到最后会发现有些东西不见了,因为它很可能已超出canvas画布以外了。
---- 形变需要在绘制图形前调用。
在这里插入图片描述

20、移动-translate

◼ translate方法,它用来移动canvas 和它的原点到一个不同的位置。
---- translate(x, y)
---- x 是左右偏移量,y是上下偏移量(无需单位)。
◼ 移动canvas 原点的好处
---- 如不使用translate方法,那么所有矩形默认都将被绘制在相同的(0,0)坐标原点。
---- translate方法可让我们任意放置图形,而不需要手工一个个调整坐标值。
◼ 移动矩形案例
---- 第一步:先保存一下canvas当前的状态
---- 第二步:在绘制图形前translate移动画布
---- 第三步:开始绘制图形,并填充颜色
在这里插入图片描述

21、旋转-rotate

◼ rotate方法,它用于以原点为中心旋转canvas,即沿着z轴旋转。
---- rotate(angle)
---- 只接受一个参数:旋转的角度(angle),它是顺时针方向,以弧度为单位的值。
---- 角度与弧度的JS表达式:弧度=( Math.PI / 180 ) * 角度,即1角度=Math.PI/180个弧度。
---- 比如:旋转90°:Math.PI / 2;旋转180°:Math.PI ;旋转360°:Math.PI * 2;旋转-90°:-Math.PI / 2;
---- 旋转的中心点始终是canvas 的原坐标点,如果要改变它,我们需要用到translate方法。
在这里插入图片描述

◼ 旋转案例
---- 第一步:先保存一下Canvas当前的状态,并确定旋转原点
---- 第二步:在绘制图形前旋转画布(坐标系会跟着旋转了)
---- 第三步:开始绘制图形,并填充颜色
在这里插入图片描述

22、缩放-scale

◼ scale方法可以缩放画布。可用它来增减图形在canvas 中的像素数目,对图形进行缩小或者放大。
---- scale(x, y)
---- x 为水平缩放因子,y 为垂直缩放因子,也支持负数。
---- 如果比1 小,会缩小图形,如果比1 大会放大图形。默认值为1。
◼ 注意事项
---- 画布初始情况下,是以左上角坐标为原点。如果参数为负实数,相当于以x 或y 轴作为对称轴镜像反转。
---- 例如,使用translate(0, canvas.height); scale(1,-1); 以y 轴作为对称轴镜像反转。
---- 默认情况下,canvas 的1 个单位为1 个像素。如果我们设置缩放因子是0.5,1 个单位就变成对应0.5 个像素,这样绘制出来的形状就会是原先的一半。同理,设置为2.0 时,1 个单位就对应变成了2 像素,绘制的结果就是图形放大了2 倍。
◼ 缩放案例实战
---- 第一步:先保存一下Canvas当前的状态,并确定缩放原点
---- 第二步:在绘制图形前缩放画布
---- 第三步:开始绘制图形,并填充颜色
在这里插入图片描述

23、Canvas动画

◼ Canvas绘图都是通过JavaScript 去操控的,如要实现一些交互性动画是相当容易的。那Canvas是如何做一些基本动画的?
---- canvas可能最大的限制就是图像一旦绘制出来,它就是一直保持那样了。
---- 如需要执行动画,不得不对画布上所有图形进行一帧一帧的重绘(比如在1秒绘60帧就可绘出流畅的动画了)。
---- 为了实现动画,我们需要一些可以定时执行重绘的方法。然而在Canvas中有三种方法可以实现:
---- 分别为setInterval 、setTimeout 和requestAnimationFrame 三种方法来定期执行指定函数进行重绘。
◼ Canvas 画出一帧动画的基本步骤(如要画出流畅动画,1s 需绘60帧):
---- 第一步:用clearRect 方法清空canvas ,除非接下来要画的内容会完全充满canvas(例如背景图),否则你需要清空所有。
---- 第二步:保存canvas 状态,如果加了canvas 状态的设置(样式,变形之类的),又想在每画一帧之时都是原始状态的话,你需要先保存一下,后面再恢复原始状态。
---- 第三步:绘制动画图形(animated shapes),即绘制动画中的一帧。
---- 第四步:恢复canvas 状态,如果已经保存了canvas 的状态,可以先恢复它,然后重绘下一帧。

24、绘制秒针-setInterval

◼ 绘制秒针动画,绘制一帧的步骤:
---- 第一步:用clearRect(x,y, w,h)方法,清空canvas。
---- 第二步:保存canvas 状态。
---- 第三步:修改canvas 状态(样式、移动坐标、旋转等)。
---- 第四步:绘制秒针图形(即绘制动画中的一帧)。
---- 第五步:恢复canvas 状态,准备重绘下一帧。
◼ setTimout定时器的缺陷
---- setTimeout定时器不是非常精准的,因为setTimeout的回调函数是放到了宏任务中等待执行。
---- 如果微任务中一直有未处理完成的任务,那么setTimeout的回调函数就有可能不会在指定时间内触发回调。
---- 如果想要更加平稳和更加精准的定时执行某个任务的话,可以使用requestAnimationFrame函数。
在这里插入图片描述

25、绘制秒针-requestAnimationFrame

◼ requestAnimationFrame函数
---- 告诉浏览器——你希望执行一个动画,并且要求浏览器在下次重绘之前调用该函数的回调函数来更新动画。
---- 该方法需要传入一个回调函数作为参数,该回调函数会在浏览器下一次重绘之前执行
---- 若想在浏览器下次重绘之前继续更新下一帧动画,那么在回调函数自身内必须再次调用requestAnimationFrame()
---- 通常每秒钟回调函数执行60 次左右,也有可能会被降低。
◼ 绘制秒针动画,绘制一帧的步骤:
---- 第一步:用clearRect(x,y, w,h)方法,清空canvas。
---- 第二步:保存canvas 状态。
---- 第三步:修改canvas 状态(样式、移动坐标、旋转等)。
---- 第四步:绘制秒针图形(即绘制动画中的一帧)。
---- 第五步:恢复canvas 状态,准备重绘下一帧。
在这里插入图片描述

26、太阳系旋转

◼ 太阳系旋转动画,绘制一帧的步骤:
---- 第一步:用clearRect(x,y, w,h)方法,清空canvas,并初始化全局样式。
---- 第二步:保存canvas 状态。
---- 第三步:绘制背景、绘制地球(绘制月球)、绘制阴影效果。
---- 第五步:恢复canvas 状态,准备重绘下一帧。
在这里插入图片描述

27、时钟

◼ 求圆上x, y的坐标:
---- 圆上x, y轴坐标实际上就是右图的( AB, BC ),AC为时钟半径
---- x= AB = cosa * AC => x = Math.cos(弧度) * R
---- y= BC = sina * AC => y = Math.sin(弧度) * R
---- 角度与弧度的JS表达式:弧度=( Math.PI / 180 ) * 角度=>弧度= 1角度对应的弧度* 角度。
---- 比如:旋转90°:弧度为Math.PI / 2;旋转180°:为Math.PI ;旋转360°:为Math.PI * 2;旋转-90°:为-Math.PI / 2;
---- 第i小时的坐标:
---- x = Math.cos( Math.PI * 2 / 12 * i ) * R
---- y = Math.sin( Math.PI * 2 / 12 * i ) * R
在这里插入图片描述

◼ 绘制时钟,绘制一帧的步骤:
---- 第一步:用clearRect(x,y, w,h)方法,清空canvas 。
---- 第二步:保存canvas 状态。
---- 第三步:绘制白背景、绘制数字、绘制时/分/秒针、绘制圆、绘制时分刻度。
---- 第四步:恢复canvas 状态,准备重绘下一帧。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/629048.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

上位机Qt应用程序与MCU板子之间的串口数据传输算法,举例1字节、2字节、4字节正负数。再加qDebug的重定向显示打印数据。

串口之间的数据传输算法 前言【1】Qt界面设计图【2】串口char型举例串口收发正数举例串口收发负数举例 【3】串口short 型举例大端序和小端序 串口收发正数举例串口收发负数举例 【4】串口int型举例串口收发正数举例串口收发负数举例串口收发正负数(简洁版推荐&…

大数据分析的Python实战指南:数据处理、可视化与机器学习【上进小菜猪大数据】

上进小菜猪,沈工大软件工程专业,爱好敲代码,持续输出干货。 引言: 大数据分析是当今互联网时代的核心技术之一。通过有效地处理和分析大量的数据,企业可以从中获得有价值的洞察,以做出更明智的决策。本文将…

Mujoco210 Ubuntu 22.04配置安装(一)

目录 .1 下载 1.1 解压 1.2 许可问题 1.3 环境配置 1.4 测试mujoco .2 安装mujoco-py 2.1 conda激活虚拟环境\或新创建一个环境 2.2 下载mujoco-py ​编辑 2.3 配置环境变量 2.4 测试mujoco-py 2.5 测试时的一些报错处理 2.5.0 command /usr/bin/gcc failed with…

Linux操作系统——第三章 基础IO

目录 接口介绍 open 文件描述符fd 0 & 1 & 2 文件描述符的分配规则 重定向 FILE 理解文件系统 inode ​编辑 理解硬链接 软链接 动态库和静态库 静态库与动态库 生成静态库 库搜索路径 生成动态库 使用动态库 运行动态库 使用外部库 接口介绍 o…

(顶刊复现)配电网两阶段鲁棒故障恢复(matlab实现)

参考文献: X. Chen, W. Wu and B. Zhang, "Robust Restoration Method for Active Distribution Networks," in IEEE Transactions on Power Systems, vol. 31, no. 5, pp. 4005-4015, Sept. 2016, doi: 10.1109/TPWRS.2015.2503426. 1.研究背景 1.1摘…

2023 Navicat for Redis 与 Navicat Premium 16.2 现已正式发布 | 释放 Redis 全部潜能

🌷🍁 博主 libin9iOak带您 Go to New World.✨🍁 🦄 个人主页——libin9iOak的博客🎐 🐳 《面试题大全》 文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~&#x1f33…

【C++】 Lambda表达式详解

▒ 目录 ▒ 🛫 问题描述环境 1️⃣ 什么是Lambda表达式Lambda 表达式的各个部分 2️⃣ 优缺点优点缺点 3️⃣ 使用场景在线C工具STL算法库STL容器中需要传递比较函数(示例失败了)多线程示例 4️⃣ Lambda表达式与函数指针的比较5️⃣ 捕获列表…

KISS复盘法

KISS复盘法 KISS复盘法是一种科学的项目复盘方法,能够把过往经验转化为实践能力,以促进下一次活动更好地展开,从而不断提升个人和团队的能力! 模型介绍 【复盘】原是围棋术语,本意是对弈者在下完一盘棋之后&#xff0…

距离保护原理

距离保护是反映故障点至保护安装处的距离,并根据距离的远近确定动作时间的一种保护。故障点距保护安装处越近,保护的动作时间就越短,反之就越长,从而保证动作的选择性。测量故障点至保护安装处的距离,实际上就是用阻抗…

Spring Boot banner详解

Spring Boot 3.x系列文章 Spring Boot 2.7.8 中文参考指南(一)Spring Boot 2.7.8 中文参考指南(二)-WebSpring Boot 源码阅读初始化环境搭建Spring Boot 框架整体启动流程详解Spring Boot 系统初始化器详解Spring Boot 监听器详解Spring Boot banner详解 自定义banner Spring …

快速排序算法的编码和优化

快速排序的基本思路是: 先通过第一趟排序,将数组原地划分为两部分,其中一部分的所有数据都小于另一部分的所有数据。原数组被划分为2份通过递归的处理, 再对原数组分割的两部分分别划分为两部分,同样是使得其中一部分…

springboot+java高校教材征订管理系统

教材管理系统从功能、数据流程、可行性、运行环境等方面进行需求分析。对教材管理系统的数据库、功能进行了详细设计。分析了主要界面设计和相关组件设计,对教材管理系统的具体实现进行了介绍。 采用Java技术,从数据库中获取数据、向数据库中写入数据&am…

Linux 4.10 将带来深远影响的三项小改变

Linux的演进永不停歇。Linus Torvalds一直在努力工作,希望能够在新的内核版本当中(4.11)融入更多变化。不过在目前的Linux 4.10中,我们同样发现了三组能够有效提升性能并实现多种前所未有功能集的变更。 Linux的演进永不停歇。Linus Torvalds一直在努力…

如何快速写出一个完整的测试用例

一、前言 测试工作中最为基础核心的内容就是设计测试用例,我们一般会认为数量越少、发现缺陷越多的用例就是好的用例。那么,怎样才能设计出好的测试用例呢?本次专题就向大家介绍如何编写一个完整且靠谱的测试用例。 二、测试用例的重要性 …

使用IPSW文件将iOS系统从Beta恢复到稳定正式版教程

起因 作为一名iOS开发者,为了拥抱新系统(手贱),将开发机升级到了最新的iOS 17 Beta版本,从而导致使用现有的Xcode无法成功配对该版本系统。故准备想方设法回滚到原先的iOS 16.5稳定版 回滚方式 若要将iOS设备回退至…

【MySQL高级篇笔记-数据库其它调优策略(中) 】

此笔记为尚硅谷MySQL高级篇部分内容 目录 一、数据库调优的措施 1、调优的目标 2、如何定位调优问题 3、调优的维度和步骤 二、优化MySQL服务器 1、优化服务器硬件 2、优化MySQL的参数 三、优化数据库结构 1、拆分表:冷热数据分离 2、增加中间表 3、增加…

CKA 01_docker部署Kubernetes 部署docker 使用kubeadm引导集群 安装Pod网络

文章目录 1. 虚拟机步骤2. Docker 部署 Kubernetes2.1 部署 docker2.1.1 环境要求2.1.2 安装 docker 引擎2.1.3 worker 节点对 master 节点免密2.1.4 设定 docker 开机自启2.1.5 打开桥接,查看桥接流量2.1.6 设定 systemd 方式管理 cgroup2.1.7 docker部署完成2.1.8…

光伏储能直流系统MATLAB仿真(PV光伏阵列+Boost DCDC变换器+负载+双向DCDC变换器+锂离子电池系统)

PV光伏阵列Boost DCDC变换器负载双向DCDC变换器锂离子电池系统 资源地址: 光伏储能直流系统MATLAB仿真(PV光伏阵列BoostDCDC变换器负载双向DCDC变换器锂离子电池系统)-Matlab文档类资源-CSDN文库 主要模块: PV光伏阵列、Boost…

这 3个Python 函数你知道吗?

动动发财的小手,点个赞吧! 作为21世纪最流行的语言之一,Python当然有很多有趣的功能值得深入探索和研究。今天将介绍其中的三个,每个都从理论上和通过实际示例进行介绍。 我想要介绍这些函数的主要原因是它们可以帮助您避免编写循…

CKA 09_Kubernetes工作负载与调度 资源调度 三类QoS request 资源需求 limit 资源限额

文章目录 1. 资源调度1.1 准备工作1.2 为什么需要 request 和 limit1.3 内存限制1.3.1 Brustable1.3.2 Guaranteed1.3.3 BestEffort1.3.4 当容器申请的资源超出 limit 和 request 1.4 CPU限制 1. 资源调度 1.1 准备工作 Kubernetes 采用 request 和 limit 两种限制类型来对资源…