zoukankan      html  css  js  c++  java
  • JS动画三剑客——setTimeout、setInterval、requestAnimationFrame

    一、前言

      前端实现动画效果主要有以下几种方法:CSS3中的transition 和 animation ,Javascript 中可以通过定时器 setTimeout、setinterval,HTML5 canvas,HTML5提供的requestAnimationFrame。本文主要分析setTimeout、setinterval、requestAnimationFrame三者的区别和他们各自的优缺点。在了解他们三个之前,我们先来看看一些相关概念。

    二、相关概念介绍

      1.屏幕刷新频率

        即图像在屏幕上更新的速度,也即屏幕上的图像每秒钟出现的次数,它的单位是赫兹(Hz)。 对于一般笔记本电脑,这个频率大概是60Hz。这个值的设定受屏幕分辨率、屏幕尺寸和显卡的影响。

      2.动画原理

        动画本质就是要让人眼看到图像被刷新而引起变化的视觉效果,这个变化要以连贯的、平滑的方式进行过渡。在屏幕每次刷新前,将图像的位置向左移动一个像素,即1px。屏幕每次刷出来的图像位置都比前一个要差1px,你就会看到图像在移动;由于我们人眼的视觉停留效应,当前位置的图像停留在大脑的印象还没消失,紧接着图像又被移到了下一个位置,因此你才会看到图像在流畅的移动,这就是视觉效果上形成的动画。

    三、setInterval

      1.运行机制

        按照指定的周期(以毫秒计)来调用函数或计算表达式。方法会不停地调用函数(当页面被隐藏或者最小化时,setInterval()仍在后台继续执行,这种动画刷新是完全没有意义的,对cpu也是极大的浪费),直到 clearInterval() 被调用或窗口被关闭。

        setinterval的执行时间不确定,参数中的时间间隔是将代码添加到异步队列中等待的时间。只有当主线程中的任务以及队列前面的任务是执行完毕,才真正开始执行动画代码。

        注:HTML5标准规定,setInterval的最短间隔时间是10毫秒,也就是说,小于10毫秒的时间间隔会被调整到10毫秒。

      2.语法

        setinterval(code, milliseconds);

        setinterval(function, milliseconds, param1, param2, ...)

    参数描述
    code/function 必需。要调用一个代码串,也可以是一个函数。
    milliseconds 必须。周期性执行或调用 code/function 之间的时间间隔,以毫秒计。
    param1, param2, ... 可选。 传给执行函数的其他参数(IE9 及其更早版本不支持该参数)。

      3.实例

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    //每三秒(3000 毫秒)弹出 "Hello":
    var myVar;
      
    function myFunction() {
        myVar = setInterval(alertFunc, 3000);
    }
      
    function alertFunc() {
        alert("Hello!");
    }

      4.清除setInterval

        clearinterval() 方法可取消由 setinterval() 函数设定的定时执行操作。参数必须是由 setinterval() 返回的 id 值。 注意: 要使用 clearinterval() 方法, 在创建执行定时操作时要使用全局变量.清除示例如下:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    var myVar = setInterval(function(){ setColor() }, 300);
      
    function setColor() {
        var x = document.body;
        x.style.backgroundColor = x.style.backgroundColor == "yellow" ? "pink" : "yellow";
    }
      
    function stopColor() {
        clearInterval(myVar);
    }

      5.缺点

       (1)setinterval()无视代码错误,如果setinterval执行的代码由于某种原因出了错,它还会持续不断地调用该代码。

       (2)setinterval无视网络延迟,由于某些原因(服务器过载、临时断网、流量剧增、用户带宽受限,等等),你的请求要花的时间远比你想象的要长。但setinterval不在乎。它仍然会按定时持续不断地触发请求,最终你的客户端网络队列会塞满调用函数。

       (3) setinterval不保证执行,与settimeout不同,并不能保证到了时间间隔,代码就准能执行。如果你调用的函数需要花很长时间才能完成,那某些调用会被直接忽略 

    四、setTimeout

      1.运行机制

        在指定的毫秒数后调用函数或计算表达式。每次函数执行的时候都会创建换一个新的定时器。在前一个定时器代码执行完之前,不会向队列插入新的定时器代码,确保不会有任何确实的间隔。并且确保在下一次定时器代码执行之前,至少要等待指定的间隔,避免了连续的运行。当方法执行完成定时器就立即停止(但是定时器还在,只不过没用了);

      2.语法(同setInterval)

      3.实例

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    //3 秒(3000 毫秒)后弹出 "Hello" :
    var myVar;
      
    function myFunction() {
        myVar = setTimeout(alertFunc, 3000);
    }
      
    function alertFunc() {
        alert("Hello!");
    }

      4.清除setTimeout

        使用cleartimeout函数,用法同clearinterval

      5.缺点

        (1)利用seTimeout实现的动画在某些低端机上会出现卡顿、抖动的现象。

        (2)settimeout的执行时间并不是确定的。在javascript中, settimeout 任务被放进了异步队列中,只有当主线程上的任务执行完以后,才会去检查该队列里的任务是否需要开始执行,因此 settimeout 的实际执行时间一般要比其设定的时间晚一些。

         (3)刷新频率受屏幕分辨率和屏幕尺寸的影响,因此不同设备的屏幕刷新频率可能会不同,而 settimeout只能设置一个固定的时间间隔,这个时间不一定和屏幕的刷新时间相同。

         (4)settimeout的执行只是在内存中对图像属性进行改变,这个变化必须要等到屏幕下次刷新时才会被更新到屏幕上。如果两者的步调不一致,就可能会导致中间某一帧的操作被跨越过去,而直接更新下一帧的图像。

    五、requestAnimationFrame(推荐使用)

      1.运行机制

        告诉浏览器——你希望执行一个动画,并且要求浏览器在下次重绘之前调用指定的回调函数更新动画。不需要设置时间间隔,是由系统的时间间隔定义的。大多数浏览器的刷新频率是60Hz(每秒钟反复绘制60次),循环间隔是1000/60,约等于16.7ms。不需要调用者指定帧速率,浏览器会自行决定最佳的帧效率。只被执行一次,这样就不会引起丢帧现象,也不会导致动画出现卡顿的问题。

      2.语法

        window.requestanimationframe(callback);

        参数callback:下一次重绘之前更新动画帧所调用的函数(即上面所说的回调函数)。

      3.实例

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    var start = null;
    var element = document.getElementById('SomeElementYouWantToAnimate');
    element.style.position = 'absolute';
     
    function step(timestamp) {
      if (!start) start = timestamp;
      var progress = timestamp - start;
      element.style.left = Math.min(progress / 10, 200) + 'px';
      if (progress < 2000) {
        window.requestAnimationFrame(step);
      }
    }
     
    window.requestAnimationFrame(step);

      4.缺点

       requestanimationframe 不管理回调函数,即在回调被执行前,多次调用带有同一回调函数的 requestanimationframe,会导致回调在同一帧中执行多次。我们可以通过一个简单的例子模拟在同一帧内多次调用 requestanimationframe 的场景:(mousemove, scroll 这类事件常见)

    1
    2
    3
    4
    5
    6
    const animation = timestamp => console.log('animation called at', timestamp)
      
    window.requestAnimationFrame(animation)
    window.requestAnimationFrame(animation)
    // animation called at 320.7559999991645
    // animation called at 320.7559999991645 

       我们用连续调用两次 requestanimationframe 模拟在同一帧中调用两次 requestanimationframe。 例子中的 timestamp 是由 requestanimationframe 传给回调函数的,表示回调队列被触发的时间。由输出可知,animation 函数在同一帧内被执行了两次,即绘制了两次动画。

      ps:解决办法

        对于这种高频发事件,一般的解决方法是使用节流函数。但是在这里使用节流函数并不能完美解决问题。因为节流函数是通过时间管理队列的,而 requestanimationframe 的触发时间是不固定的,在高刷新频率的显示屏上时间会小于 16.67ms,页面如果被推入后台,时间可能大于 16.67ms。

        完美的解决方案是通过 requestanimationframe 来管理队列,其思路就是保证 requestanimationframe 的队列里,同样的回调函数只有一个。示例代码如下:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    const onScroll = e => {
        if (scheduledAnimationFrame) { return }
      
        scheduledAnimationFrame = true
        window.requestAnimationFrame(timestamp => {
            scheduledAnimationFrame = false
            animation(timestamp)
        })
    }
    window.addEventListener('scroll', onScroll)

      5.与setTimeout和setInterval的区别

        (1)requestanimationframe会把每一帧中的所有dom操作集中起来,在一次重绘或回流中就完成,并且重绘或回流的时间间隔紧紧跟随浏览器的刷新频率

        (2)在隐藏或不可见的元素中,requestanimationframe将不会进行重绘或回流,这当然就意味着更少的cpu、gpu和内存使用量

        (3)requestanimationframe是由浏览器专门为动画提供的api,在运行时浏览器会自动优化方法的调用,并且如果页面不是激活状态下的话,动画会自动暂停,有效节省了cpu开销

      6.兼容性封装

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    if(!window.requestAnimationFrame) {
     window.requestAnimationFrame = (window.webkitRequestAnimationFrame ||
     window.mozRequestAnimationFrame ||
     window.oRequestAnimationFrame ||
     window.msRequestAnimationFrame ||
     function(callback) {
      var self = this, start, finish;
      return window.setTimeout(function() {
       start = +new Date();
       callback(start);
       finish = +new Date();
       self.timeout = 1000/60 - (finish - start);
      }, self.timeout);
     });
    }

      代码解析:

        这段代码先检查了 window.requestanimationframe 函数的定义是否存在。如果不存在,就遍历已知的各种浏览器实现并替代该函数。如果还是找不到一个与浏览器相关的实现,它最终会采用基于javascript定时器的动画以每秒60帧的间隔调用settimeout函数。

        mozrequestanimationframe() 会接收一个时间码(从1970年1月1日起至今的毫秒数),表示下一次重绘的实际发生时间。这样, mozrequestanimationframe() 就会根据这个时间码设定将来的某个时刻进行重绘。

         但是 webkitrequestanimationframe() 和 msrequestanimationframe() 不会给回调函数传递时间码,因此无法知道下一次重绘将发生在什么时间。 如果要计算两次重绘的时间间隔,firefox中可以使用既有的时间码,而在chrome和ie则可以使用不太精确地date()对象。

      7.清除动画

        cancelAnimationFrame(动画名) ,类似clearTimeout函数

        

     六、总结

      1.执行次数:setInterval执行多次,setTimeout、requestAnimationframe执行一次

      2.性能:setTimeout会出现丢帧、卡顿现象,setInterval会出现调用丢失情况,requestAnimationframe不会出现这些问题,页面未激活时不会执行动画,减少了大量cpu消耗

      3.兼容性问题:setInterval,setTimeout在IE浏览器中不支持参数传递,能够在大多数浏览器中正常使用。而requestAnimationframe不兼容IE10以下

    七、面试题

      1.setTimeout中的this指向问题

    1
    2
    3
    4
    5
    6
    7
    8
    var i = 0;
    const o = {
        i: 1;
        fn: function(){
            console.log(this.i);
        }
    }
    setTimeout(o.fn, 1000); //执行后会打印出什么

        错误思路:setTimeout执行,调用对象O的fn函数,由于调用者是对象O,那么this也指向了对象O,又对象O中有属性i,则会打印出1。

        正解:因为setTimeout是window对象的方法,传入o.fn只是将o.fn这个函数传给了setTimeout,仍然是window对象在调用。上面代码执行的正确结果是0,是因为定义了全局变量i为0。如果没有定义,则会输出undefined。

        ps:如果这里不是setTimeout执行这个函数,而是o.fn(),那么会输出1。

      2.执行下面的代码,控制台如何输出

    (function () {
        setTimeout(function () {
            alert(2);
        }, 0);
    
        alert(1);
    })()  

        先弹出的应该是1,而不是你以为“立即执行”的2。 settimeout,setinterval都存在一个最小延迟的问题,虽然你给的delay值为0,但是浏览器执行的是自己的最小值。html5标准是4ms,但并不意味着所有浏览器都会遵循这个标准,包括手机浏览器在内,这个最小值既有可能小于4ms也有可能大于4ms。在标准中,如果在settimeout中嵌套一个settimeout, 那么嵌套的settimeout的最小延迟为10ms。

      3.执行下面的代码,控制台输出什么

    1
    2
    3
    4
    5
    for (var i = 1; i <= 5; i++) {
      setTimeout(function timer() {
        console.log(i)
      }, i * 1000)
    }

        输出结果大家都只是会是5个6,由于JavaScript是单线程的,按顺序执行,setTimeout是异步函数,它会将 timer 函数放到任务队列中,而此时会先将循环执行完毕再执行 timer 函数,因此当执行 timer 函数时 i 已经等于6了,所以最终会输出5个6

        ps:解决办法有三种,我只贴代码了

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    //闭包
    for (var i = 1; i <= 5; i++) {
      (function(j) {
        setTimeout(function timer() {
          console.log(j)
        }, j * 1000)
      })(i)
    }
     
    //给setTimeout传参
    //方式一 IE不支持
    for (var i = 1; i <= 5; i++) {
      setTimeout(
        function timer(j) {
          console.log(j)
        },
        i * 1000,
        i
      )
    }//方式二for (var i = 1; i <= 5; i++) {
    1
    2
      (function(i){    setTimeout(function(){        console.log(i)    },i * 1000)  })(i)
    }
    1
    2
    3
    4
    5
    6
    7
    //ES6 let
     
    for (let i = 1; i <= 5; i++) {
      setTimeout(function timer() {
        console.log(i)
      }, i * 1000)
    }

      4.使用settimeout代替setinterval进行间歇调用

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    var executeTimes = 0;
    var intervalTime = 500;
    var intervalId = null;
     
    // 放开下面的注释运行setInterval的Demo
    intervalId = setInterval(intervalFun,intervalTime);
    // 放开下面的注释运行setTimeout的Demo
    // setTimeout(timeOutFun,intervalTime);
     
    function intervalFun(){
        executeTimes++;
        console.log("doIntervalFun——"+executeTimes);
        if(executeTimes==5){
            clearInterval(intervalId);
        }
    }
     
    function timeOutFun(){
        executeTimes++;
        console.log("doTimeOutFun——"+executeTimes);
        if(executeTimes<5){
            setTimeout(arguments.callee,intervalTime);
        }
    }

      代码比较简单,我们只是在settimeout的方法里面又调用了一次settimeout,就可以达到间歇调用的目的。 setinterval间歇调用,是在前一个方法执行前,就开始计时,比如间歇时间是500ms,那么不管那时候前一个方法是否已经执行完毕,都会把后一个方法放入执行的序列中。这时候就会发生一个问题,假如前一个方法的执行时间超过500ms,加入是1000ms,那么就意味着,前一个方法执行结束后,后一个方法马上就会执行,因为此时间歇时间已经超过500ms了。

      5.利用settimeout来实现setinterval

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    function interval(func, w, t){
        var interv = function(){
            if(typeof t === "undefined" || t-- > 0){
                setTimeout(interv, w);
                try{
                    func.call(null);
                }
                catch(e){
                    t = 0;
                    throw e.toString();
                }
            }
        };
     
        setTimeout(interv, w);
    };

    参考文档:https://blog.csdn.net/weixin_34204057/article/details/89009605

         http://www.luyixian.cn/javascript_show_149688.aspx

         https://juejin.im/post/5c89fe42e51d455bb15c1ed1

         https://www.cnblogs.com/icctuan/p/12103697.html  

         

     
  • 相关阅读:
    求职方法论
    测试经验与教训_学习笔记
    测试架构师修炼之道_学习笔记
    Jmeter测试oracle
    Jmeter 非UI界面jmx脚本不能正常退出
    Jmeter参数化的理解
    jmeter 测试并发
    Jmeter测试数据库
    pytorch runtime error: CUDNN_STATUS_MAPPING_ERROR
    Python/pytorch 切换国内源/AttributeError: module 'torch.jit' has no attribute 'unused'/not a trusted or secure host
  • 原文地址:https://www.cnblogs.com/linybo/p/13214959.html
Copyright © 2011-2022 走看看