不会吧,不会吧,现在都2020年了不会还真人有人不知道JS的rAF吧???

rAF

简介

rAF是requestAnimationFrame的简称;

我们先从字面意思上理解requestAnimationFrame「request - 请求」「Animation - 动画」「Frame - 帧率;框架」rAF难道是JS的动画框架???,结果显而易见并不是。但确实rAF和动画有关系

我们先来看一下MDN官网对的requestAnimationFrame解释:

window.requestAnimationFrame() 告诉浏览器——你希望执行一个动画,并且要求浏览器在下次重绘之前调用指定的回调函数更新动画。该方法需要传入一个回调函数作为参数,该回调函数会在浏览器下一次重绘之前执行

浏览器兼容性

requestAnimationFrame兼容IE10及以上,这时候有人会有疑问,怎么才到IE10啊,但其实我们最常使用的CSS3 animation属性也是IE10之后才有的,在IE9之前想要实现动画基本使用的是setTimeout/setInterval实现

作用及用法

requestAnimationFrame简称rAF,它是浏览器全局对象window的一个方法。

相比于setTimeout的在固定时间后执行对应的动画函数,rAF用于指示浏览器在下一次重新绘制屏幕图像时, 执行其提供的回调函数。

这也是rAF的最大优势–它能够保证我们的动画函数的每一次调用都对应着一次屏幕重绘,从而避免setTimeout通过时间定义动画频率,与屏幕刷新频率不一致导致的丢帧。

详细用法

requestAnimationFrame语法如下:


  1. window.requestAnimationFrame(callback)

「参数;callback」
下一次重绘之前更新动画帧所调用的函数(即上面所说的回调函数)。该回调函数会被传入DOMHighResTimeStamp参数,该参数与performance.now()的返回值相同,它表示requestAnimationFrame()开始去执行回调函数的时刻。

「返回值」一个 long 整数,请求 ID ,是回调列表中唯一的标识。是个非零值,没别的意义。你可以传这个值给 window.cancelAnimationFrame() 以取消回调函数。

DOMHighResTimeStamp 指的是一个double类型,用于存储毫秒级的时间值。这种类型可以用来描述离散的时间点或者一段时间(两个离散时间点之间的时间差)。

performance.now()方法返回一个精确到毫秒的DOMHighResTimeStamp 。


它的实际常见用法类似于setTimeout,只是不需要设置时间间隔而已。如下:


  1. const element = document.getElementById('some-element-you-want-to-animate');
  2. let start;
  3. function step(timestamp) {
  4. // timestamp回调函数传入的`DOMHighResTimeStamp`参数,也就是存储毫秒级的时间值
  5. if (start === undefined)
  6. start = timestamp;
  7. const elapsed = timestamp - start;
  8. //这里使用`Math.min()`确保元素刚好停在200px的位置。
  9. element.style.transform = 'translateX(' + Math.min(0.1 * elapsed, 200) + 'px)';
  10. if (elapsed < 2000) { // 在两秒后停止动画
  11. window.requestAnimationFrame(step);
  12. }
  13. }
  14. window.requestAnimationFrame(step);

上述代码的作用在每一次屏幕显示图像的更新中,都将元素向左移动1px,停在200px位置上。

实际使用示例

「上才艺,E G M,E G M E G M E G M」

我们以在3000毫秒内移动1500px距离的动画为例

setTimeout的实现方式

以下代码通过setTimeout每10毫秒为间隔时间改变一次元素的位置以实现元素的动画效果, 当然,可以通过改变这个间隔时间来微调动画效果,可是你永远没有办法确定最优方案,因为它总会和刷新频率存在交叉。


  1. <div id="div" style="width:100px; height:100px; background-color:#000; position: absolute;left:0; top:0;">
  2. </div>
  3. <script type="text/javascript">
  4. let divEle = document.getElementById("div");
  5. const distance = 1500; // 需要移动的距离
  6. const timeCount = 3000; // 需要使用的时间
  7. const intervalTime = 10; // 设置间隔时间为10ms
  8. let runCount = timeCount / intervalTime; // 相除得到运行次数
  9. let moveValue = distance / runCount; // 每次运行移动的距离
  10. function handler() {
  11. let left = parseInt(divEle.style.left);
  12. if(left >= distance) {
  13. // 当距离左侧的距离超出需要移动的距离停止
  14. return;
  15. }
  16. divEle.style.left = left + moveValue;
  17. window.setTimeout(handler, intervalTime);
  18. }
  19. window.setTimeout(handler, intervalTime);
  20. </script>

requestAnimationFrame的实现方式

「从setTimeout切换到 requestAnimationFrame很容易,因为它们都安排了一个回调。对于连续动画,在调用动画函数之后再次调用requestAnimationFrame。」

request 会把每一帧中的所有DOM操作集中起来,在一次重绘或回流中就完成(这点很像虚拟DOM不是~),并且重绘或回流的时间间隔紧紧跟随浏览器的刷新频率,这样就不会出现过度渲染的问题,保证了流畅的需求以及浏览器的完美渲染。


  1. <div id="div" style="width:100px; height:100px; background-color:#000; position: absolute;left:0; top:0;">
  2. </div>
  3. <script type="text/javascript">
  4. let divEle = document.getElementById("div");
  5. const distance = 1500; // 需要移动的距离
  6. const timeCount = 3000; // 需要使用的时间
  7. function handler( time ) {
  8. // time为rAF返回的毫秒级时间单位,当time的大于timeCount的值则停止
  9. // time理论上是从 1 开始到timeCount定义的3000,
  10. if(time > timeCount) {
  11. time = timeCount;
  12. }
  13. // 这句代码的作用是 time理论上是从 1 至 3000
  14. // 当到达3000的时候,time * distance / timeCount得到的一定是distance的值1500
  15. divEle.style.left = time * distance / timeCount;
  16. window.requestAnimationFrame( handler ); // 循环调用,渲染完成会停止
  17. }
  18. window.requestAnimationFrame( handler );
  19. </script>

requestAnimationFrame的优点

为什么不使用settimeout?

setTimeout通过设定一个时间间隔来不断的更新屏幕图像,从而完成动图。
它的优点是可控性高,可以进行编码式的动画效果实现。

setTimeout缺点:

  1. 「造成无用的函数运行开销:」

也就是过度绘制,同时因为更新图像的频率和屏幕的刷新重绘制步调不一致,会产生丢帧,在低性能的显示器动画看起来就会卡顿。

  1. 「当网页标签或浏览器置于后台不可见时,仍然会执行,造成资源浪费」

  2. 「API本身达不到毫秒级的精确:」

如果使用 setTimeout或者setInterval 那么需要我们制定时间 假设给予 (1000/60)理论上就可以完成60帧速率的动画。所以事实是浏览器可以“强制规定时间间隔的下限(clamping th timeout interval)”,一般浏览器所允许的时间再5-10毫秒,也就是说即使你给了某个小于10的数,可能也要等待10毫秒。

  1. 「浏览器不能完美执行:」

当动画使用10ms的settimeout绘制动画时,您将看到一个时序不匹配,如下所示。

我们的显示屏一般是「16.7ms(即60FPS)的显示频率」,上图的第一行代表大多数监视器上显示的「16.7ms显示频率」,上图的第二行代表「10ms的典型setTimeout」。由于在显示刷新间隔之前发生了另一个绘制请求,因此无法绘制每次的第三个绘制(红色箭头指示)。这种透支会导致动画断断续续,「因为每三帧都会丢失」。计时器分辨率的降低也会对电池寿命产生负面影响,并降低其他应用程序的性能。

如果使用requestAnimationFrame可以解决setTimeout的丢帧问题,因为它使应用程序时通知(且仅当)的浏览器需要更新页面显示,渲染时间由系统处理。因此,应用程序与浏览器绘画间隔完全一致,并且仅使用适当数量的资源。

requestAnimationFrame的好处

相比于setTimeout的在固定时间后执行对应的动画函数,requestAnimationFrame用于指示浏览器在下一次重新绘制屏幕图像时, 执行其提供的回调函数。

  • 「使浏览器画面的重绘和回流与显示器的刷新频率同步」它能够保证我们的动画函数的每一次调用都对应着一次屏幕重绘,从而避免setTimeout通过时间定义动画频率,与屏幕刷新频率不一致导致的丢帧。

  • 「节省系统资源,提高性能和视觉效果」在页面被置于后台或隐藏时,会自动的停止,不进行函数的执行,当页面激活时,会重新从上次停止的状态开始执行,因此在性能开销上也会相比setTimeout小很多。

兼容问题

目前的时间点上,几乎所有的浏览器现行版本都支持了requestAnimationFrame函数。但在一部分浏览器上还需要加上兼容性前缀。
下面这是比较全面的方法用来使requestAnimation兼容各浏览器:


  1. (function() {
  2. var lastTime = 0;
  3. var vendors = ['webkit', 'moz']; // 浏览器前缀
  4. // 当window.requestAnimationFrame不存在时执行for循环,添加前缀
  5. for(var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
  6. window.requestAnimationFrame = window[vendors[x] + 'RequestAnimationFrame'];
  7. window.cancelAnimationFrame = window[vendors[x] + 'CancelAnimationFrame'] ||
  8. window[vendors[x] + 'CancelRequestAnimationFrame'];
  9. }
  10. //当添加前缀后依旧不存在,则使用setTimeout替代
  11. if (!window.requestAnimationFrame) {
  12. window.requestAnimationFrame = function(callback, element) {
  13. var currTime = new Date().getTime();
  14. var timeToCall = Math.max(0, 16.7 - (currTime - lastTime));
  15. var id = window.setTimeout(function() {
  16. callback(currTime + timeToCall);
  17. }, timeToCall);
  18. lastTime = currTime + timeToCall;
  19. return id;
  20. };
  21. }
  22. if (!window.cancelAnimationFrame) {
  23. window.cancelAnimationFrame = function(id) {
  24. clearTimeout(id);
  25. };
  26. }
  27. }());

然后,我们就可以以使用setTimeout的感觉使用requestAnimationFrame方法制作动画啦!

相关链接

结尾

如有疑问,可在下方留言,会第一时间进行回复

谢谢你愿意花时间阅读这篇文章,希望可以对你有所帮助!

我曾踏足山巅,也曾跌落谷底,两者都让我受益良多。

2020已经过去五分之四了,你确定还不来了解一下JS的rAF?的更多相关文章

  1. [TPYBoard - Micropython] 五分种学会用TPYBoard - GPS 制作短信群发机

    转载请注明:@小五义 http://www.cnblogs.com/xiaowuyi 欢迎加入讨论群 64770604   一.什么是TPYBoard-GPS TPYBoard-GPS又称TPYBoa ...

  2. [TPYBoard - Micropython 之会python就能做硬件 9] 五分种学会用TPYBoard V102 制作避障小车(升级版)

    转载请注明:@小五义 http://www.cnblogs.com/xiaowuyi 欢迎加入讨论群 64770604 感谢山东萝卜电子科技公司授权   一.实验器材 1.TPYboard V102板 ...

  3. LocalDateTime查找最近的五分钟点

    /** * 最近的五分钟 * @param dateTime * @return */ public static LocalDateTime getNear5(LocalDateTime dateT ...

  4. 《find技巧》-“linux命令五分系列”之一

    一天一个命令,做个记录, 我要成大神,哈哈哈 本原创文章属于<Linux大棚>博客. 博客地址为http://roclinux.cn. 文章作者为roc 希望您能通过捐款的方式支持Linu ...

  5. laravel报错:SQLSTATE[23000]: Integrity constraint violation: 1062 Duplicate entry '0' for key 'PRIMARY' (SQL: insert into `cart` (`uid`, `gid`, `gname`, `price`) values (3, 21, 夏季日系复古工装短袖衬衫男士印花潮流宽松五分

    原因:要操作的数据表id没有设置自增,导致出现id为0的情况 解决方法:给该数据表的id字段设置自增

  6. C#语句2——循环语句(for循环与for循环嵌套)

    循环:反复执行某段代码. 循环四要素:初始条件,循环条件,循环体,状态改变. for(初始条件;循环条件;状态改变) { 循环体 } break ——中断循环,跳出整个循环 continue——停止本 ...

  7. Java:过去、未来的互联网编程之王

    Java对你而言是什么?一门你大学里学过的语言?一个IT行业的通用语言?你相信Java已经为下一次互联网爆炸做好了准备么?Java 一方面为嵌入式计算做了增强,而另一方面为实时应用做了精简,本文将介绍 ...

  8. 2014年全球SEO行业调查报告

    前言: 1.该调查报告是MOZ每两年一度针对SEO行业的数据分析报告. 2.随着SEO的进化,该报告已不仅仅是SEO行业,今年的调查数据更多分析网络营销行业,可以称作"网络营销行业调查报告& ...

  9. HttpWebRequest提高效率之连接数,代理,自动跳转,gzip请求等设置问题

    先设置4个: [csharp] webrequest.ServicePoint.Expect100Continue = false; //是否使用 Nagle 不使用 提高效率 webrequest. ...

随机推荐

  1. C#知识点:操作XML

    XML是什么就不用说了文本标记语言. 主要纪录如何对XML文件进行增删改查. Xml的操作类都存在System.xml命名空间下面. 应用型的直接上代码 using System; using Sys ...

  2. TKE基于弹性网卡直连Pod的网络负载均衡

    前言 Kubernetes在集群接入层设计并提供了两种原生资源Service和Ingress,分别负责四层和七层的网络接入层配置. 传统的做法是创建Ingress或LoadBalancer类型的Ser ...

  3. python中绑定码云仓库

    1.File——Settings——Version Control——Git——输入git安装路径下bin下的git.exe路径——点击后面的Test测试一下,弹出版本点击ok即可 2.点击工具栏中的 ...

  4. Tomcat 第一篇:源码导入 IDEA 编辑器

    1 引言 做 Java 的同学应该都见过上面这只名字叫 Tomcat 的猫,毕竟这只猫在过去和现在都是全球最流行的 Web 容器之一. 很有意思的一件事儿是从我接触这只猫开始,从来不知道它的中文名字是 ...

  5. [Binder深入学习二]Binder驱动——基础数据结构二

    Userspace和KernelSpace进行交互时,大部分命令是通过 ioctl 实现的,在这个过程中,最重要的一个便是 BINDER_WRITE_READ 命令了. #define BINDER_ ...

  6. 如何在 PyPI安装python的软件包?

    安装软件包 本节介绍如何安装Python的基本知识.包裹. 需要注意的是,这个上下文中的“包”一词被用作分布(即要安装的一组软件),而不是指包装在Python源代码中导入(即模块的容器).Python ...

  7. 传值&传值引用

    转自http://www.cnblogs.com/androidsuperman/p/9012320.html 首先对传值和传引用要有个基本的概念 传值:传递的是值的副本.方法中对副本的修改,不会影响 ...

  8. Hadoop演进与Hadoop生态

    1.了解对比Hadoop不同版本的特性,可以用图表的形式呈现. (1)0.20.0~0.20.2: Hadoop的0.20分支非常稳定,虽然看起来有些落后,但是经过生产环境考验,是 Hadoop历史上 ...

  9. 腾讯云ClickHouse如何实现自动化的数据均衡?

    ​一.引言 ClickHouse 是一个用于联机分析( OLAP )的列式数据库管理系统( DBMS ).它于 2016 年以 Apache 2.0 协议开源,以优秀的查询性能,深受广大大数据工程师欢 ...

  10. 总结一下,selenium 自动化流程如下

    自动化程序调用Selenium 客户端库函数(比如点击按钮元素) 客户端库会发送Selenium 命令 给浏览器的驱动程序 浏览器驱动程序接收到命令后 ,驱动浏览器去执行命令 浏览器执行命令 浏览器驱 ...