1. 简介
原生 JavaScript 中,我们可以通过 setTimeout() 或是 setInterval() 来不断更新元素状态以实现动画效果。要看到流畅的动画效果,就需要在更新元素状态时以一定的频率进行,我们先来了解一下“帧”的概念。
以下是百度百科是关于“帧”的说明:
为了更好地说明帧的概念,我们先来看看电影播放的基本原理。
在放映电影的过程中,画面被一幅幅地放映在银幕上。画幅移开时,光线就被遮住,幕上便出现短暂的黑暗;每放映一个画幅后,幕上就黑暗一次。但这一次次极短暂的黑暗,被人的视觉生理现象“视觉暂留”所弥补。人眼在观察景物时,光信号传入大脑神经需经过一段短暂时间,光的作用结束时,视觉也不立即消失。视觉的这一现象称为“视觉暂留”。当电影画面换幅频率达到每秒15幅~30幅时,观众便见不到黑暗的间隔了,这时人“看到”的就是运动的事物,这就是电影的基本原理。这里的一幅画面就是电影的一帧,实际上就是电影胶片中的一格。
帧——就是影像动画中最小单位的单幅影像画面。一帧就是一副静止的画面,连续的帧就形成动画,如电视图象等。我们通常说帧数,简单地说,就是在1秒钟时间里传输的图片的帧数,也可以理解为图形处理器每秒钟能够刷新几次,通常用 FPS(Frames Per Second)表示。每一帧都是静止的图象,快速连续地显示帧便形成了运动的假象。高的帧率可以得到更流畅、更逼真的动画。每秒钟帧数 (fps) 愈多,所显示的动作就会愈流畅。
了解了帧的概念,我们再回顾一下 《浏览器是如何工作的?》 。
浏览器中网页的生成过程大致可以分成五步:
这几个步骤中,第 1 到 3 步执行速度非常快,但第 4、5 步就比较耗时了。
网页在生成时至少会渲染一次,在用户访问过程中,还会不断地重新渲染,重新渲染就需要重新生成布局(reflow,也称重排)和重新绘制(repaint,也称重绘)。修改 DOM、修改样式表或是用户事件(如页面滚动、改变窗口大小等)都会导致页面重新渲染,reflow 和 repaint 不断被触发,这在用户访问过程中是不可避免的。
因为 reflow 和 repaint 非常耗时,所以重新渲染也是导致网页性能低下的根本原因,要提高网页性能,就是要降低 reflow 和 repaint 的频率和成本,尽量少触发重新渲染。
但很多时候,密集的重新渲染又是不可避免的,比如 scroll 事件响应程序的处理、网页动画效果等。
网页动画的每一帧都是一次重新渲染。每秒低于 24 帧的动画,人眼就能捕获到停顿,动画效果就感觉有卡顿,一般网页动画,需要达到 30 至 60 fps 才能比较流畅。网页动画和电影动画的帧率又有些差别,可参考 https://www.zhihu.com/question/21081976 继续了解,不详细说明。
大多数显示器的刷新频率是 60Hz(1Hz = 1次/秒),所以,如果网页动画能够做到每秒 60 帧,就会跟显示器同步刷新,达到最佳的视觉效果。这意味着,一秒之内进行 60 次重新渲染,每次重新渲染的时间不能超过 16.7 毫秒,这也是我们在使用 setTimeout() 或 setInterval() 定时器实现动画效果时推荐的时间间隔。
但是,使用 setTimeout() 和 setInterval() 绘制的动画并没有为 Web 开发人员提供有效的方法来规划动画的图形计时器。这导致了动画过度绘制,浪费 CPU 周期以及消耗额外的电能等问题。而且,即使看不到网站,特别是当网站使用背景选项卡中的页面或浏览器已最小化时,动画都会频繁出现(新版浏览器中对此也有优化,可参见 翻译:setInterval与requestAnimationFrame的时间间隔测试 )。
使用时间间隔 10ms(过度绘制)的计时器绘制动画时,计时与显示器刷新频率可能不匹配,如下所示:
上面一行表示大多数监视器上显示的 16.7ms 显示频率,而下面一行表示 10ms setTimeout()。每个第三个图形都无法绘制(由红色箭头指示),因为在显示器刷新间隔之前发生了其他绘制请求。这种过度绘制的情况会导致动画断续显示,因为所有第三帧都会丢失。这种计时器定时时间降低也会对电池使用寿命造成负面影响,并会降低其它应用的性能。
requestAnimationFrame() 方法可以解决丢失帧的问题,因为它使应用能够在浏览器需要更新页面显示时获得通知,它用来在页面重绘之前,通知浏览器调用一个指定的函数,以满足开发者操作动画的需求。因此,使用 requestAnimationFrame(),应用可与浏览器的绘制时间间隔保持完全一致,并且仅使用适量的资源,可实现性能上的优化。
2. 使用 window.requestAnimationFrame
requestAnimationFrame() 方法原理其实跟 setTimeout 差不多,通过递归调用同一方法来不断更新画面以达到动起来的效果,但它优于 setTimeout 的地方在于它是由浏览器专门为动画提供的API,在运行时浏览器会自动优化方法的调用,并且如果页面不是激活状态下的话,动画会自动暂停,有效节省了CPU开销。
var id = window.requestAnimationFrame(callback);callback 参数为回调函数,在每次重新绘制动画时调用该函数,该回调函数有一个参数,表示当前时间距离开始触发 requestAnimationFrame 的回调的时间。
该方法返回一个长整型非 0 值,作为唯一的标识符。可以像停止定时器 clearTimeout() 一样,将返回值传递到 window.cancelAnimationFrame() 中,取消一个先前通过调用 window.requestAnimationFrame() 方法添加到计划中的动画帧请求。
下面是一个旋转动画的例子,元素每一帧旋转 1 度,为了快速查找页面元素及绑定事件处理,使用了 jQuery:
<script type="text/javascript" src="js/jquery-3.1.1.min.js"></script> <script type="text/javascript"> $(function(){ // 点击按钮,开启旋转动画 $("#btn").click(function(){ update(); // 角度 var degrees = 0; // 动画方法 function update() { if (degrees >= 360) return; $("#box").css("transform", "rotate("+ (++degrees) +"deg)"); window.requestAnimationFrame(update); // 递归调用实现动画效果 </script> function update() { if (progress >= 100) return; $("#progressbar").css("width", ++progress * 2).text(progress+"%"); window.requestAnimationFrame(update); // 递归调用实现动画效果 </script>更为具体的兼容性大家可以通过 caniuse 查询。
由上图可以看出,在老版本的浏览器中,requestAnimationFrame() 方法不被支持,为了让代码能够有更好的浏览器兼容性,即在旧版本的浏览器上也能运行不报错,我们可以写一段代码让浏览器在不支持 requestAnimationFrame() 的情况下使用 setTimeout() 以达到类似效果,把这样的代码称为 Polyfill(垫片),通俗点来说,就是代码备胎的意思。
以下借鉴 paulirish 发布在 GitHub Gist 上的代码片段 requestAnimationFrame polyfill,用于在不支持 requestAnimationFrame 的浏览器中回退到 setTimeout 来实现:
// http://paulirish.com/2011/requestanimationframe-for-smart-animating/ // http://my.opera.com/emoller/blog/2011/12/20/requestanimationframe-for-smart-er-animating // requestAnimationFrame polyfill by Erik Möller. fixes from Paul Irish and Tino Zijdel // MIT license (function() { var lastTime = 0; var vendors = ['ms', 'moz', 'webkit', 'o']; for(var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) { window.requestAnimationFrame = window[vendors[x]+'RequestAnimationFrame']; window.cancelAnimationFrame = window[vendors[x]+'CancelAnimationFrame'] || window[vendors[x]+'CancelRequestAnimationFrame']; if (!window.requestAnimationFrame) window.requestAnimationFrame = function(callback, element) { var currTime = new Date().getTime(); var timeToCall = Math.max(0, 16 - (currTime - lastTime)); var id = window.setTimeout(function() { callback(currTime + timeToCall); }, timeToCall); lastTime = currTime + timeToCall; return id; if (!window.cancelAnimationFrame) window.cancelAnimationFrame = function(id) { clearTimeout(id); }());4. 小结
虽然 CSS3 中的 transition 或 animation 动画也能实现与 requestAnimationFrame 一样的绘制原理,但 CSS3 动画也不是适用于所有的属性,如 scrollTop 值,CSS3 就无能为力了。同时,CSS3 支持的动画效果有限,如果要实现某些特殊的缓动效果,还得使用 requestAnimationFrame 或 setTimeout 来实现。而 setTimeout 又可能存在过度绘制问题,浪费 CPU 资源或消耗更多额外的电池电能,所以使用 requestAnimationFrame 来优化是很有必要的。
基于脚本的动画的计时控制(“requestAnimationFrame”)
网页性能管理详解:浅谈chrome-Timeline及window.requestAnimationFrame()方法
MDN window.requestAnimationFrame