写一个 JavaScript 框架:比 setTimeout 更棒的定时执行

这个系列是关于一个开源的客户端框架,叫做 NX。在这个系列里,我主要解释一下写该框架不得不克服的主要困难。如果你对 NX 感兴趣可以参观我们的 主页

这个系列包含以下几个章节:

  1. 项目结构
  2. 定时执行 (当前章节)
  3. 沙箱代码评估
  4. 数据绑定介绍
  5. 数据绑定与 ES6 代理
  6. 自定义元素
  7. 客户端路由

异步代码执行

你可能比较熟悉 Promiseprocess.nextTick()setTimeout(),或许还有requestAnimationFrame() 这些异步执行代码的方式。它们内部都使用了事件循环,但是它们在精确计时方面有一些不同。

在这一章里,我将解释它们之间的不同,然后给大家演示怎样在一个类似 NX 这样的先进框架里面实现一个定时系统。不用我们重新做一个,我们将使用原生的事件循环来达到我们的目的。

事件循环

事件循环甚至没有在 ES6 规范里提到。JavaScript 自身只有任务(Job)和任务队列(job queue)。更加复杂的事件循环是在 NodeJS 和 HTML5 规范里分别定义的,因为这篇是针对前端的,我会在详细说明后者。

事件循环可以被看做某个条件的循环。它不停的寻找新的任务来运行。这个循环中的一次迭代叫做一个滴答(tick)。在一次滴答期间执行的代码称为一次任务(task)。


  1. while (eventLoop.waitForTask()) {
  2. eventLoop.processNextTask()
  3. }

任务是同步代码,它可以在循环中调度其它任务。一个简单的调用新任务的方式是setTimeout(taskFn)。不管怎样, 任务可能有很多来源,比如用户事件、网络或者 DOM 操作。

任务队列

更复杂一些的是,事件循环可以有多个任务队列。这里有两个约束条件,相同任务源的事件必须在相同的队列,以及任务必须按插入的顺序进行处理。除此之外,浏览器可以做任何它想做的事情。例如,它可以决定接下来处理哪个任务队列。


  1. while (eventLoop.waitForTask()) {
  2. const taskQueue = eventLoop.selectTaskQueue()
  3. if (taskQueue.hasNextTask()) {
  4. taskQueue.processNextTask()
  5. }
  6. }

用这个模型,我们不能精确的控制定时。如果用 setTimeout()浏览器可能决定先运行完其它几个队列才运行我们的队列。

微任务队列

幸运的是,事件循环还提供了一个叫做微任务(microtask)队列的单一队列。当前任务结束的时候,微任务队列会清空每个滴答里的任务。


  1. while (eventLoop.waitForTask()) {
  2. const taskQueue = eventLoop.selectTaskQueue()
  3. if (taskQueue.hasNextTask()) {
  4. taskQueue.processNextTask()
  5. }
  6. const microtaskQueue = eventLoop.microTaskQueue
  7. while (microtaskQueue.hasNextMicrotask()) {
  8. microtaskQueue.processNextMicrotask()
  9. }
  10. }

最简单的调用微任务的方法是 Promise.resolve().then(microtaskFn)。微任务按照插入顺序进行处理,并且由于仅存在一个微任务队列,浏览器不会把时间弄乱了。

此外,微任务可以调度新的微任务,它将插入到同一个队列,并在同一个滴答内处理。

绘制Rendering

最后是绘制Rendering调度,不同于事件处理和分解,绘制并不是在单独的后台任务完成的。它是一个可以运行在每个循环滴答结束时的算法。

在这里浏览器又有了许多自由:它可能在每个任务以后绘制,但是它也可能在好几百个任务都执行了以后也不绘制。

幸运的是,我们有 requestAnimationFrame(),它在下一个绘制之前执行传递的函数。我们最终的事件模型像这样:


  1. while (eventLoop.waitForTask()) {
  2. const taskQueue = eventLoop.selectTaskQueue()
  3. if (taskQueue.hasNextTask()) {
  4. taskQueue.processNextTask()
  5. }
  6. const microtaskQueue = eventLoop.microTaskQueue
  7. while (microtaskQueue.hasNextMicrotask()) {
  8. microtaskQueue.processNextMicrotask()
  9. }
  10. if (shouldRender()) {
  11. applyScrollResizeAndCSS()
  12. runAnimationFrames()
  13. render()
  14. }
  15. }

现在用我们所知道知识来创建定时系统!

利用事件循环

和大多数现代框架一样,NX 也是基于 DOM 操作和数据绑定的。批量操作和异步执行以取得更好的性能表现。基于以上理由我们用 Promises、 MutationObservers 和 requestAnimationFrame()

我们所期望的定时器是这样的:

  1. 代码来自于开发者
  2. 数据绑定和 DOM 操作由 NX 来执行
  3. 开发者定义事件钩子
  4. 浏览器进行绘制

步骤 1

NX 寄存器对象基于 ES6 代理 以及 DOM 变动基于MutationObserver (变动观测器)同步运行(下一节详细介绍)。 它作为一个微任务延迟直到步骤 2 执行以后才做出反应。这个延迟已经在Promise.resolve().then(reaction) 进行了对象转换,并且它将通过变动观测器自动运行。

步骤 2

来自开发者的代码(任务)运行完成。微任务由 NX 开始执行所注册。 因为它们是微任务,所以按序执行。注意,我们仍然在同一个滴答循环中。

步骤 3

开发者通过 requestAnimationFrame(hook) 通知 NX 运行钩子。这可能在滴答循环后发生。重要的是,钩子运行在下一次绘制之前和所有数据操作之后,并且 DOM 和 CSS 改变都已经完成。

步骤 4

浏览器绘制下一个视图。这也有可能发生在滴答循环之后,但是绝对不会发生在一个滴答的步骤 3 之前。

牢记在心里的事情

我们在原生的事件循环之上实现了一个简单而有效的定时系统。理论上讲它运行的很好,但是还是很脆弱,一个轻微的错误可能会导致很严重的 BUG。

在一个复杂的系统当中,最重要的就是建立一定的规则并在以后保持它们。在 NX 中有以下规则:

  1. 永远不用 setTimeout(fn, 0) 来进行内部操作
  2. 用相同的方法来注册微任务
  3. 微任务仅供内部操作
  4. 不要干预开发者钩子运行时间

规则 1 和 2

数据反射和 DOM 操作将按照操作顺序执行。这样只要不混合就可以很好的延迟它们的执行。混合执行会出现莫名其妙的问题。

setTimeout(fn, 0) 的行为完全不可预测。使用不同的方法注册微任务也会发生混乱。例如,下面的例子中 microtask2 不会正确地在 microtask1 之前运行。


  1. Promise.resolve().then().then(microtask1)
  2. Promise.resolve().then(microtask2)

规则 3 和 4

分离开发者的代码执行和内部操作的时间窗口是非常重要的。混合这两种行为会导致不可预测的事情发生,并且它会需要开发者了解框架内部。我想很多前台开发者已经有过类似经历。

原文发布时间为:2017-11-26

本文来自合作伙伴“Linux中国”

时间: 2024-08-11 22:48:48

写一个 JavaScript 框架:比 setTimeout 更棒的定时执行的相关文章

写一个JavaScript框架:比setTimeout更棒的定时执行

这是 JavaScript 框架系列的第二章.在这一章里,我打算讲一下在浏览器里的异步代码不同执行方式.你将了解定时器和事件循环之间的不同差异,比如 setTimeout 和 Promises. 这个系列是关于一个开源的客户端框架,叫做 NX.在这个系列里,我主要解释一下写该框架不得不克服的主要困难.如果你对 NX 感兴趣可以参观我们的 主页. 这个系列包含以下几个章节: 项目结构 定时执行 (当前章节) 沙箱代码评估 数据绑定介绍 数据绑定与 ES6 代理 自定义元素 客户端路由 异步代码执行

你可能不需要一个 JavaScript 框架(一)

你可能不需要一个 JavaScript 框架 You (probably) don't need a JavaScript framework 我并不打算写一篇类似于<为何 JavaScript 社区如此不堪>的文章那是因为我觉得真没必要.只是我认为事情本来就很简单,而且以"就地取材"的方式去做事情也确实十分有趣.下面我就为您一一娓娓道来,介绍到底 Web API 和原生 DOM 有多么强大和多么简单. I'm not going to post yet another r

java 线程 sql qualz-如何写一个定时任务要每月的1号01:00执行这个sql

问题描述 如何写一个定时任务要每月的1号01:00执行这个sql 1C insert into test_dept_monthhours(yearmonthbranchsubtotal) select yearmonthbranchsum(dm)+sum(sm)+sum(smt)+sum(sup)+sum(ltnc) from ehruser.HO_DEPT_MONTHHOURS@traininglink where rownum<5 group by branchyearmonth 解决方案

你可能不需要一个 JavaScript 框架(二)

 解决方案 Solution 你需要的,还是 JavaScript 和 Web API.我希望阁下可以看看 Web API (detailed),快速浏览一下然后回到本文. What you need is vanilla JavaScript and the Web API. I want you to take a look at the Web API (detailed). Scimm through it for a couple of minutes and come back. 看

写一个ORM框架的第一步(Apache Commons DbUtils)

新一次的内部提升开始了,如果您想写一个框架从Apache Commons DbUtils开始学习是一种不错的选择,我们先学习应用这个小"框架"再把源代码理解,然后写一个属于自己的ORM框架不是梦. 一.简介 DbUtils是Apache下commons工具集中的一个小工具,它主要是对JDBC封装的ORM小工具,简化了JDBC的操作.之所以把它称之为工具而不是框架,是因为它和其他的ORM框架还是由很大的区别(例如Hibernate).DbUtils并不支持所谓的聚合关联映射.缓存机制.实

我现在要用C#写一个程序实现两个数据库之间的定时查询和insert

问题描述 求如果用c#应该怎么写定时的时间是每天的19:00:00,求Demo 解决方案 解决方案二:我抛个砖,如果要求不严格可以使用定时器.每1分钟30秒啥的取一次时间,是这个时间久执行最今本的有3种计时器和一种基本计时方法,他们的用途不一样,不知道你用的什么方式,所以不给你推荐.你可以查找计时器或计时方式进行学习.这种情况我一般用计划任务,好处是不用维护程序,到点自己就跑了解决方案三:while(true){try{if(DateTime.Now.ToString("HH:mm:ss&quo

怎么写一个javascript 实现一个纠结的页面生成? 请大家指点

问题描述 在主页 由一个新闻标题 点击弹出新闻内容的html页面.现在有一个前台控制器 需要大家指点一下javascript的写法.望大家帮助. 解决方案 下面的代码拷贝另存为xx.html<script>function openUrl(url){window.open(url,null,"height=600,width=800,status=yes,toolbar=no,menubar=no,location=no,scrollbars=yes,resizable=yes&qu

如何写一个框架(转)

  说明:作者也没写过什么框架,只是分享一些自己的理解,抛砖引玉罢了.如果你写过一些框架可能会产生一些共鸣欢迎讨论,如果你正在写或正打算写一个框架可能会给你一些启发.本文以为较长可能会分多个篇博客来写,现在能想到的是主要分为步骤.模式两部分.如果你觉得好,按一个推荐举手之劳让更多的人可以看到.   步骤   定位   所谓定位就是回答几个问题,我出于什么目的要写一个框架,我的这个框架是干什么的,有什么特性适用于什么场景,我的这个框架的用户对象是谁,他们会怎么使用,框架由谁维护将来怎么发展等等.

九个用于移动APP开发的顶级JavaScript框架

从技术上讲,iOS.Android和Windows Phone上的移动app使用了不同的编程语言进行编码.iOS app使用Objective-C,Android app使用Java,而Windows Phone app使用.NET.但是,掌握一定量的JavaScript.CSS和HTML知识,你就可以构建超棒的移动app.因此,在本博客中,我们将讨论用于开发移动app的顶级JavaScript框架. 对于Web开发而言,JavaScript是一个有前途的编程语言,并且在不久的将来它将依然在这个