React Scheduler 源码详解(1)

栏目: 服务器 · 发布时间: 5年前

内容简介:自从因此,本篇文章就是来干这个事情的,从源码角度来一步步阐述虽然说标题是

自从 react 16 出来以后, react fiber 相关的文章层出不穷,但大多都是讲解 fiber 的数据结构,以及组件树的 diff 是如何由递归改为循环遍历的。对于 time slicing 的描述一般都说利用了 requestIdleCallback 这个api来做调度,但对于任务如何调度却很难找到详细的描述。

因此,本篇文章就是来干这个事情的,从源码角度来一步步阐述 React Scheduler 是怎么实现任务调度的。

虽然说标题是 React Scheduler ,但本文的内容跟 react 是不相关的,因为任务调度器其实跟 react 是没有关系的,它只是描述怎么在合适的时机去执行一些任务,也就是说你即使没有 react 基础也可以进行本文的阅读,如果你是框架作者,也可以借鉴这个 scheduler 的实现,在自己的框架里来进行任务调度。

  • 本文讲解的是 react v16.7.0 版本的源码,请注意时效性。
  • 源码路径 Scheduler.js

2、基础知识

接下来先来了解一下阅读本文需要知道的一些基础知识。

1、 window.performance.now

这个是浏览器内置的时钟,从页面加载开始计时,返回到当前的总时间,单位 ms 。意味着你在打开页面第10分钟在控制台调用这个方法,返回的数字大概是 600000(误)。

2、 window.requestAnimationFrame

  • 这个方法应该很常见了,它让我们可以在下一帧开始时调用指定的函数。它的执行是是跟随系统的刷新频率的。 requestAnimationFrame 方法接收一个参数,即要执行的回调函数。这个回调函数会默认地传入一个参数,即从打开页面到回调函数被触发时的时间长度,单位为毫秒。

  • 可以理解为系统在调用回调前立马执行了一下 performance.now() 传给了回调当参数。这样我们就可以在执行回调的时候知道当前的执行时间了。

    requestAnimationFrame(function F(t) {
           console.log(t, '===='); //会不断打印执行回调的时间,如果刷新频率为60Hz,则相邻的t间隔时间大约为1000/60 = 16.7ms
           requestAnimationFrame(F)
       })
    复制代码
  • requestAnimationFrame 有个特点,就是当页面处理未激活的状态下, requestAnimationFrame 会停止执行;当页面后面再转为激活时, requestAnimationFrame 又会接着上次的地方继续执行。

3、 window.MessageChannel

这个接口允许我们创建一个新的消息通道,并通过它的两个 MessagePort(port1,port2) 属性发送数据。 示例代码如下

var channel = new MessageChannel();
    var port1 = channel.port1;
    var port2 = channel.port2;
    port1.onmessage = function(event){
        console.log(event.data)  // someData
    }
    port2.postMessage('someData')
复制代码

这里有一点需要注意, onmessage 的回调函数的调用时机是在一帧的paint完成之后。据观察 vuenextTick 也是用 MessageChannel 来做 fallback 的(优先用 setImmediate )。

react scheduler 内部正是利用了这一点来在一帧渲染结束后的剩余时间来执行任务的

4、 链表

先默认大家对链表有个基本的认识。没有的话自己去补一下知识。

这里要介绍的是双向循环链表

  • 双向链表是指每个节点有 previousnext 两个属性来分别指向前后两个节点。
  • 循环的意思是,最后一个节点的next指向第一个节点,而第一个节点的 previous 指向最后一个节点,形成一个环形的 人体蜈蚣
  • 我们还需要用一个变量firstNode来存储第一个节点。
  • 下面以一个具体例子来讲一下双向循环链表的插入和删除操作,假设有一群人需要按照年龄进行排队,小孩站前边,大人站后边。在一个过程内会不断有人过来,我们需要把他插到正确的位置。删除的话只考虑每次把排头的人给去掉。
//person的类型定义
    interface Person {
        name : string  //姓名
        age : number  //年龄,依赖这个属性排序
        next : Person  //紧跟在后面的人,默认是null
        previous : Person //前面相邻的那个人,默认是null
    }
    var firstNode = null; //一开始链表里没有节点
    
    //插入的逻辑
    function insertByAge(newPerson:Person){
        if(firstNode = null){
        
        //如果 firstNode为空,说明newPerson是第一个人,  
        //把它赋值给firstNode,并把next和previous属性指向自身,自成一个环。
          firstNode = newPerson.next = newPerson.previous = newPerson;
          
        } else { //队伍里有人了,新来的人要找准自己的位置
        
             var next = null; //记录newPerson插入到哪个人前边
             var person = firstNode; // person 在下边的循环中会从第一个人开始往后找
             
             do {
                  if (person.age > newPerson.age) {
                  //如果person的年龄比新来的人大,说明新来的人找到位置了,他恰好要排在person的前边,结束
                    next = person;
                    break;
                  }
                  //继续找后面的人
                  node = node.next;
            } while (node !== firstNode); //这里的while是为了防止无限循环,毕竟是环形的结构
            
            if(next === null){ //找了一圈发现 没有person的age比newPerson大,说明newPerson应该放到队伍的最后,也就是说newPerson的后面应该是firstNode。
                next = firstNode;
            }else if(next === firstNode){ //找第一个的时候就找到next了,说明newPerson要放到firstNode前面,这时候firstNode就要更新为newPerson
                firstNode = newPerson
            }
            
            //下面是newPerson的插入操作,给next及previous两个人的前后链接都关联到newPerson
            var previous = next.previous;
            previous.next = next.previous = newPerson; 
            newPerson.next = next;
            newPerson.previous = previous;
        }
        //插入成功
    }
    
    //删除第一个节点
    function deleteFirstPerson(){
        if(firstNode === null) return; //队伍里没有人,返回
        
        var next = firstNode.next; //第二个人
        if(firstNode === next) {
            //这时候只有一个人
            firstNode = null;
            next = null;
        } else {
            var lastPerson = firstNode.previous; //找到最后一个人
            firstNode = lastPerson.next = next; //更新新的第一人
            next.previout = lastPerson; //并在新的第一人和最后一人之间建立连接
        }
        
    }
    
复制代码

由于 react16 内大量利用了链表来记录数据,尤其 react scheduler 内对任务的操作使用了双向循环链表结构。所以理解了上述的代码,对于理解 react 对任务的调度就会比较容易了。

3、正文

注:为了梳理整体的运行流程,下面的示例代码有可能会在源码基础上有少量删减

0、 几个方法,下文不再赘述

``
    getCurrentTime = function() {
        return performance.now();
        //如果不支持performance,利用 Date.now()做fallback
    }
```
复制代码

1、任务优先级

react内对任务定义的优先级分为5种,数字越小优先级越高

var ImmediatePriority = 1;  //最高优先级
   var UserBlockingPriority = 2; //用户阻塞型优先级
   var NormalPriority = 3; //普通优先级
   var LowPriority = 4; // 低优先级
   var IdlePriority = 5; // 空闲优先级
复制代码

这5种优先级依次对应5个过期时间

// Max 31 bit integer. The max integer size in V8 for 32-bit systems.
   // Math.pow(2, 30) - 1
   var maxSigned31BitInt = 1073741823;

   // 立马过期 ==> ImmediatePriority
   var IMMEDIATE_PRIORITY_TIMEOUT = -1;
   // 250ms以后过期
   var USER_BLOCKING_PRIORITY = 250;
   //
   var NORMAL_PRIORITY_TIMEOUT = 5000;
   //
   var LOW_PRIORITY_TIMEOUT = 10000;
   // 永不过期
   var IDLE_PRIORITY = maxSigned31BitInt;
复制代码

每个任务在添加到链表里的时候,都会通过 performance.now() + timeout 来得出这个任务的过期时间,随着时间的推移,当前时间会越来越接近这个过期时间,所以过期时间越小的代表优先级越高。如果过期时间已经比当前时间小了,说明这个任务已经过期了还没执行,需要立马去执行( asap )。

上面的 maxSigned31BitInt ,通过注释可以知道这是 32 位系统 V8 引擎里最大的整数。 react 用它来做 IdlePriority 的过期时间。

据粗略计算这个时间大概是 12.427 天。也就是说极端情况下你的网页 tab 如果能一直开着到12天半,任务才有可能过期。

2、 function scheduleCallback()

  • 代码里的方法叫做 unstable_scheduleCallback ,意思是当前还是不稳定的,这里就以 scheduleCallback 作名字。
  • 这个方法的作用就是把任务以过期时间作为优先级进行排序,过程类似上文双向循环链表的操作过程。

下面上代码

function scheduleCallback(callback, options? : {timeout:number} ) {
       //to be coutinued
   }
复制代码

这个方法有两个入参,第一个是要执行的 callback ,暂时可以理解为一个任务。第二个参数是可选的,可以传入一个超时时间来标识这个任务过多久超时。如果不传的话就会根据上述的任务优先级确定过期时间。

//这是一个全局变量,代表当前任务的优先级,默认为普通
  var currentPriorityLevel = NormalPriority
  
  function scheduleCallback(callback, options? : {timeout:number} ) {
      var startTime = getCurrentTime()
      if (
          typeof options === 'object' &&
          options !== null &&
          typeof options.timeout === 'number'
        ){
          //如果传了options, 就用入参的过期时间
          expirationTime = startTime + options.timeout;
        } else {
          //判断当前的优先级
          switch (currentPriorityLevel) {
            case ImmediatePriority:
              expirationTime = startTime + IMMEDIATE_PRIORITY_TIMEOUT;
              break;
            case UserBlockingPriority:
              expirationTime = startTime + USER_BLOCKING_PRIORITY;
              break;
            case IdlePriority:
              expirationTime = startTime + IDLE_PRIORITY;
              break;
            case LowPriority:
              expirationTime = startTime + LOW_PRIORITY_TIMEOUT;
              break;
            case NormalPriority:
            default:
              expirationTime = startTime + NORMAL_PRIORITY_TIMEOUT;
          }
        }
        
        //上面确定了当前任务的截止时间,下面创建一个任务节点,
        var newNode = {
          callback, //任务的具体内容
          priorityLevel: currentPriorityLevel, //任务优先级
          expirationTime, //任务的过期时间
          next: null, //下一个节点
          previous: null, //上一个节点
        };
      //to be coutinued
  }
复制代码

上面的代码根据入参或者当前的优先级来确定当前 callback 的过期时间,并生成一个真正的任务节点。接下来就要把这个节点按照 expirationTime 排序插入到任务的链表里边去。

// 代表任务链表的第一个节点
   var firstCallbackNode = null;
   
   function scheduleCallback(callback, options? : {timeout:number} ) {
       ...
       var newNode = {
           callback, //任务的具体内容
           priorityLevel: currentPriorityLevel, //任务优先级
           expirationTime, //任务的过期时间
           next: null, //下一个节点
           previous: null, //上一个节点
       };
       // 下面是按照 expirationTime 把 newNode 加入到任务队列里。参考基础知识里的person排队的例子
       
       if (firstCallbackNode === null) {
           firstCallbackNode = newNode.next = newNode.previous = newNode;
           ensureHostCallbackIsScheduled(); //这个方法先忽略,后面讲
       } else {
           var next = null;
           var node = firstCallbackNode;
           do {
             if (node.expirationTime > expirationTime) {
               next = node;
               break;
             }
             node = node.next;
           } while (node !== firstCallbackNode);

       if (next === null) {
         next = firstCallbackNode;
       } else if (next === firstCallbackNode) {
         firstCallbackNode = newNode;
         ensureHostCallbackIsScheduled(); //这个方法先忽略,后面讲
       }
   
       var previous = next.previous;
       previous.next = next.previous = newNode;
       newNode.next = next;
       newNode.previous = previous;
     }
   
     return newNode;
       
   }
复制代码
  • 上面的逻辑除了 ensureHostCallbackIsScheduled 就是前面讲的双向循环链表的插入逻辑。
  • 到这里一个新进来的任务如何确定过期时间以及如何插入现有的任务队列就讲完了。
  • 到这里就会不禁产生一个疑问,我们把任务按照过期时间排好顺序了,那么何时去执行任务呢?
  • 答案是有两种情况,1是当添加第一个任务节点的时候开始启动任务执行,2是当新添加的任务取代之前的节点成为新的第一个节点的时候。因为1意味着任务从无到有,应该 立刻启动。2意味着来了新的优先级最高的任务,应该停止掉之前要执行的任务,重新从新的任务开始执行。
  • 上面两种情况就对应 ensureHostCallbackIsScheduled 方法执行的两个分支。所以我们现在应该知道, ensureHostCallbackIsScheduled 是用来在合适的时机去启动任务执行的。
  • 到底什么是合适的时机?可以这么描述,在每一帧绘制完成之后的空闲时间。这样就能保证浏览器绘制每一帧的频率能跟上系统的刷新频率,不会掉帧。

接下来就需要实现这么一个功能,如何在合适的时机去执行一个function。

3 requestIdleCallback pollyfill

现在请暂时忘掉上面那段任务队列相关的事情,来思考如何在浏览器每一帧绘制完的空闲时间来做一些事情。

答案可以是 requestIdleCallback ,但由于某些原因,react团队放弃了这个api,转而利用 requestAnimationFrameMessageChannel pollyfill 了一个 requestIdleCallback

1、 function requestAnimationFrameWithTimeout()

首先介绍一个超强的函数,代码如下

var requestAnimationFrameWithTimeout = function(callback) {
      rAFID = requestAnimationFrame(function(timestamp) {
        clearTimeout(rAFTimeoutID);
        callback(timestamp);
      });
      rAFTimeoutID = setTimeout(function() {
        cancelAnimationFrame(rAFID);
        callback(getCurrentTime());
      }, 100);
    }
复制代码

这段代码什么意思呢?

  • 当我们调用 requestAnimationFrameWithTimeout 并传入一个 callback 的时候,会启动一个 requestAnimationFrame 和一个 setTimeout ,两者都会去执行 callback 。但由于 requestAnimationFrame 执行优先级相对较高,它内部会调用 clearTimeout 取消下面定时器的操作。所以在页面 active 情况下的表现跟 requestAnimationFrame 是一致的。

  • 到这里大家应该明白了,一开始的基础知识里说了, requestAnimationFrame 在页面切换到未激活的时候是不工作的,这时 requestAnimationFrameWithTimeout 就相当于启动了一个 100ms 的定时器,接管任务的执行工作。这个执行频率不高也不低,既能不影响cpu能耗,又能保证任务能有一定效率的执行。

  • 下面我们暂时先认为 requestAnimationFrameWithTimeout 等价于 requestAnimationFrame

(不知不觉篇幅已经这么长了,今天先写到这里吧,下次有机会再更)


以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

剑指Offer

剑指Offer

何海涛 / 电子工业出版社 / 2012-1 / 45.00元

《剑指Offer:名企面试官精讲典型编程题》剖析了50个典型的程序员面试题,从基础知识、代码质量、解题思路、优化效率和综合能力五个方面系统整理了影响面试的5个要点。全书分为7章,主要包括面试的流程,讨论面试流程中每一环节需要注意的问题;面试需要的基础知识,从编程语言、数据结构及算法三方面总结了程序员面试的知识点;高质量的代码,讨论影响代码质量的3个要素(规范性、完整性和鲁棒性),强调高质量的代码除......一起来看看 《剑指Offer》 这本书的介绍吧!

URL 编码/解码
URL 编码/解码

URL 编码/解码

SHA 加密
SHA 加密

SHA 加密工具

XML 在线格式化
XML 在线格式化

在线 XML 格式化压缩工具