队列是 Node.js 中用于有效处理异步操作的一项重要技术。

在本文中,我们将深入研究 Node.js 中的队列:它们是什么,它们如何工作(通过事件循环)以及它们的类型。

Node.js 中的队列是什么?

队列是 Node.js 中用于组织异步操作的数据结构。这些操作以不同的形式存在,包括HTTP请求、读取或写入文件操作、流等。

在 Node.js 中处理异步操作非常具有挑战性。

HTTP 请求期间可能会出现不可预测的延迟(或者更糟糕的可能性是没有结果),具体取决于网络质量。尝试用 Node.js 读写文件时也有可能会产生延迟,具体取决于文件的大小。

类似于计时器和其他的许多操作,异步操作完成的时间也有可能是不确定的。

在这些不同的延迟情况之下,Node.js 需要能够有效地处理所有这些操作。

Node.js 无法处理基于 first-start-first-handle (先开始先处理)或 first-finish-first-handle (先结束先处理)的操作。

之所以不能这样做的一个原因是,在一个异步操作中可能还会包含另一个异步操作。

为第一个异步过程留出空间意味着必须先要完成内部异步过程,然后才能考虑队列中的其他异步操作。

有许多情况需要考虑,因此最好的选择是制定规则。这个规则影响了事件循环和队列在 Node.js 中的工作方式。

让我们简要地看一下 Node.js 是怎样处理异步操作的。

调用栈,事件循环和回调队列

调用栈被用于跟踪当前正在执行的函数以及从何处开始运行。当一个函数将要执行时,它会被添加到调用堆栈中。这有助于 JavaScript 在执行函数后重新跟踪其处理步骤。

回调队列是在后台操作完成时把回调函数保存为异步操作的队列。它们以先进先出(FIFO)的方式工作。我们将会在本文后面介绍不同类型的回调队列。

请注意,Node.js 负责所有异步活动,因为 JavaScript 可以利用其单线程性质来阻止产生新的线程。

在完成后台操作后,它还负责向回调队列添加函数。 JavaScript 本身与回调队列无关。同时事件循环会连续检查调用栈是否为空,以便可以从回调队列中提取一个函数并添加到调用栈中。事件循环仅在执行所有同步操作之后才检查队列。

那么,事件循环是按照什么样的顺序从队列中选择回调函数的呢?

首先,让我们看一下回调队列的五种主要类型。

回调队列的类型

IO 队列(IO queue)

IO操作是指涉及外部设备(如计算机的硬盘、网卡等)的操作。常见的操作包括读写文件操作、网络操作等。这些操作应该是异步的,因为它们留给 Node.js 处理。

JavaScript 无法访问计算机的内部设备。当执行此类操作时,JavaScript 会将其传输到 Node.js 以在后台处理。

完成后,它们将会被转移到 IO 回调队列中,来进行事件循环,以转移到调用栈中执行。

计时器队列(Timer queue)

每个涉及 Node.js 计时器功能的操作(如 setTimeout()setInterval())都是要被添加到计时器队列的。

请注意,JavaScript 语言本身没有计时器功能。它使用 Node.js 提供的计时器 API(包括 setTimeout )执行与时间相关的操作。所以计时器操作是异步的。无论是 2 秒还是 0 秒,JavaScript 都会把与时间相关的操作移交给 Node.js,然后将其完成并添加到计时器队列中。

例如:

setTimeout(function() {console.log('setTimeout');}, 0)console.log('yeah')# 返回
yeah
setTimeout

在处理异步操作时,JavaScript 会继续执行其他操作。只有在所有同步操作都已被处理完毕后,事件循环才会进入回调队列。

微任务队列(Microtask queue)

该队列分为两个队列:

  • 第一个队列包含因 process.nextTick 函数而延迟的函数。

事件循环执行的每个迭代称为一个 tick(时间刻度)。

process.nextTick 是一个函数,它在下一个 tick (即事件循环的下一个迭代)执行一个函数。微任务队列需要存储此类函数,以便可以在下一个 tick 执行它们。

这意味着事件循环必须继续检查微任务队列中的此类函数,然后再进入其他队列。

  • 第二个队列包含因 promises 而延迟的函数。

如你所见,在 IO 和计时器队列中,所有与异步操作有关的内容都被移交给了异步函数。

但是 promise 不同。在 promise 中,初始变量存储在 JavaScript 内存中(你可能已经注意到了<Pending>)。

异步操作完成后,Node.js 会将函数(附加到 Promise)放在微任务队列中。同时它用得到的结果来更新 JavaScript 内存中的变量,以使该函数不与 <Pending> 一起运行。

以下代码说明了 promise 是如何工作的:

let prom = new Promise(function (resolve, reject) {// 延迟执行setTimeout(function () {return resolve("hello");}, 2000);});console.log(prom);// Promise { <pending> }prom.then(function (response) {console.log(response);});// 在 2000ms 之后,输出// hello

关于微任务队列,需要注意一个重要功能,事件循环在进入其他队列之前要反复检查并执行微任务队列中的函数。例如,当微任务队列完成时,或者说计时器操作执行了 Promise 操作,事件循环将会在继续进入计时器队列中的其他函数之前参与该 Promise 操作。

因此,微任务队列比其他队列具有最高的优先级。

检查队列(Check queue)

检查队列也称为即时队列(immediate queue)。IO 队列中的所有回调函数均已执行完毕后,立即执行此队列中的回调函数。setImmediate 用于向该队列添加函数。

例如:

const fs = require('fs');
setImmediate(function() {console.log('setImmediate');
})
// 假设此操作需要 1ms
fs.readFile('path-to-file', function() {console.log('readFile')
})
// 假设此操作需要 3ms
do...while...

执行该程序时,Node.js 把 setImmediate 回调函数添加到检查队列。由于整个程序尚未准备完毕,因此事件循环不会检查任何队列。

因为 readFile 操作是异步的,所以会移交给 Node.js,之后程序将会继续执行。

do while 操作持续 3ms。在这段时间内,readFile 操作完成并被推送到 IO 队列。完成此操作后,事件循环将会开始检查队列。

尽管首先填充了检查队列,但只有在 IO 队列为空之后才考虑使用它。所以在 setImmediate 之前,将 readFile 输出到控制台。

关闭队列(Close queue)

此队列存储与关闭事件操作关联的函数。

包括以下内容:

  • 流关闭事件,在关闭流时发出。它表示不再发出任何事件。
  • http关闭事件,在服务器关闭时发出。

这些队列被认为是优先级最低的,因为此处的操作会在以后发生。

你肯sing不希望在处理 promise 函数之前在 close 事件中执行回调函数。当服务器已经关闭时,promise 函数会做些什么呢?

队列顺序

微任务队列具有最高优先级,其次是计时器队列,I/O队列,检查队列,最后是关闭队列。

回调队列的例子

让我们通过一个更复杂的例子来说明队列的类型和顺序:

const fs = require("fs");// 假设此操作需要 2ms
fs.writeFile('./new-file.json', '...', function() {console.log('writeFile')
})// 假设这需要 10ms 才能完成
fs.readFile("./file.json", function(err, data) {console.log("readFile");
});// 不需要假设,这实际上需要 1ms
setTimeout(function() {console.log("setTimeout");
}, 1000);// 假设此操作需要 3ms
while(...) {...
}setImmediate(function() {console.log("setImmediate");
});// 解决 promise 需要 4 ms
let promise = new Promise(function (resolve, reject) {setTimeout(function () {return resolve("promise");}, 4000);
});
promise.then(function(response) {console.log(response)
})console.log("last line");

程序流程如下:

  • 在 0 毫秒时,程序开始。
  • 在 Node.js 将回调函数添加到 IO 队列之前,fs.writeFile 在后台花费 2 毫秒。

fs.readFile takes 10ms at the background before Node.js adds the callback function to the IO queue.

  • 在 Node.js 将回调函数添加到 IO 队列之前,fs.readFile 在后台花费 10 毫秒。
  • 在 Node.js 将回调函数添加到计时器队列之前,setTimeout 在后台花费 1ms。
  • 现在,while 操作(同步)需要 3ms。在此期间,线程被阻止(请记住 JavaScript 是单线程的)。
  • 同样在这段时间内,setTimeoutfs.writeFile 操作完成,并将它们的回调函数分别添加到计时器和 IO 队列中。

现在的队列是:

// queues
Timer = [function () {console.log("setTimeout");},
];
IO = [function () {console.log("writeFile");},
];

setImmediate 将回调函数添加到 Check 队列中:

js
// 队列
Timer...
IO...
Check = [function() {console.log("setImmediate")}
]

在将 promise 操作添加到微任务队列之前,需要花费 4ms 的时间在后台进行解析。

最后一行是同步的,因此将会立即执行:

# 返回
"last line"

因为所有同步活动都已完成,所以事件循环开始检查队列。由于微任务队列为空,因此它从计时器队列开始:

// 队列
Timer = [] // 现在是空的
IO...
Check...# 返回
"last line"
"setTimeout"

当事件循环继续执行队列中的回调函数时,promise 操作完成并被添加到微任务队列中:

// 队列Timer = [];Microtask = [function (response) {console.log(response);},];IO = []; // 当前是空的Check = []; // 当前是在 IO 的后面,为空# results"last line""setTimeout""writeFile""setImmediate"

几秒钟后,readFile 操作完成,并添加到 IO 队列中:

// 队列Timer = [];Microtask = []; // 当前是空的IO = [function () {console.log("readFile");},];Check = [];# results"last line""setTimeout""writeFile""setImmediate""promise"

最后,执行所有回调函数:

// 队列Timer = []Microtask = []IO = [] // 现在又是空的Check = [];# results"last line""setTimeout""writeFile""setImmediate""promise""readFile"

这里要注意的三点:

  • 异步操作取决于添加到队列之前的延迟时间。并不取决于它们在程序中的存放顺序。
  • 事件循环在每次迭代之继续检查其他任务之前,会连续检查微任务队列。
  • 即使在后台有另一个 IO 操作(readFile),事件循环也会执行检查队列中的函数。这样做的原因是此时 IO 队列为空。请记住,在执行 IO 队列中的所有的函数之后,将会立即运行检查队列回调。

总结

JavaScript 是单线程的。每个异步函数都由依赖操作系统内部函数工作的 Node.js 去处理。

Node.js 负责将回调函数(通过 JavaScript 附加到异步操作)添加到回调队列中。事件循环会确定将要在每次迭代中接下来要执行的回调函数。

了解队列如何在 Node.js 中工作,使你对其有了更好的了解,因为队列是环境的核心功能之一。 Node.js 最受欢迎的定义是 non-blocking(非阻塞),这意味着异步操作可以被正确的处理。都是因为有了事件循环和回调队列才能使此功能生效。

原文:https://blog.logrocket.com/a-deep-dive-into-queues-in-node-js/

js 延迟几秒执行_深入研究 Node.js 的回调队列相关推荐

  1. js读取http chunk流_极简 Node.js入门 教程双工流

    点击上方蓝字关注我们 小编提示: 本文是由 ICBU 的谦行小哥哥出品,我们会持续发出极简 Node.js入门 教程,敬请期待哦,文中有比较多的演示代码建议横屏阅读 双工流就是同时实现了 Readab ...

  2. js 延迟几秒执行ifarme_Node.js调试之llnode篇

    概述:本文简单介绍一下如何使用llnode来分析Node应用挂掉之后产生的Core dump文件,帮助您快速定位问题. Core dump 首先介绍一下什么是Core dump,根据维基百科,它是应用 ...

  3. js 延迟几秒执行ifarme_js设定延迟时间的函数

    1.如果想要在执行一个js函数之前延迟一段时间应该怎么做? 答:"setTimeout('update()',1000);" 其中update()函数就是延迟后执行的函数,后面的时 ...

  4. js延迟2秒执行事件

    有时候,我们在做修改回显数据时,就需要默认触发一些事件,但是由于数据没有很快从服务器中取回,所以就有延迟执行js事件 setTimeout(function () {// 这里就是处理的事件 }, 2 ...

  5. js 延迟几秒执行ifarme_延时加载JavaScript代码提高速度_javascript技巧 -

    ...值: none 0 e1. 利用animation属性实现循环间的延时执行实例教程简介:先来介绍一下animation定义和用法,animation 属性是一个简写属性,用于设置六个动画属性:2 ...

  6. js 延迟几秒执行ifarme_用JS控制iframe里的页面,做到3秒自动换一个。

    展开全部 假定iframe的id为ifr 下面是js程序 var times = 10;  //循环次数32313133353236313431303231363533e4b893e5b19e3133 ...

  7. android 延迟2秒执行_每天30秒让你更懂汽车(10自动变速器2执行装置)

    本篇内容 848字,阅读时长约 55 秒 执行装置包括换挡离合器.换挡制动器和单向离合器等. 控制装置根据汽车不同行驶条件,分别在执行装置中建立或卸除油压, 从而得到自动变速器的不同挡位. 1.换挡离 ...

  8. 超简单JS延迟5秒加载方法代码

    JS延迟5秒加载方法 setTimeout( function(){ //add your code }, 5 * 1000 ); //延迟5000毫米 5000等于5秒哦

  9. node.js使用手册_权威的Node.js手册

    node.js使用手册 Note: you can get a PDF, ePub, or Mobi version of this handbook for easier reference, or ...

最新文章

  1. 基于连通域字符分割的流程_基于改进连通域算法的车牌字符分割方法
  2. 数据为什么要可视化?如何可视化?
  3. linux文件怎么操作,linux文件操作学习1
  4. android 启动器开发,Android启动器(Launcher)开发详解
  5. mysql服务不能启动的几大原因
  6. 基于PCL的ICP及其变种算法实现
  7. 【leetcode】75. Sort Colors
  8. 找出二维数组中每行的最小数但最小数的列不能相同
  9. android 系统(143)---Android实现App版本自动更新
  10. java接口_Java接口
  11. 基于行跟踪的ROWDEPENDENCIES ORA_ROWSCN信息
  12. 马哥运维班第一周作业
  13. oracle10g rac导出ocr,Oracle RAC OCR磁盘故障快速恢复方法
  14. TestBed静态测试步骤
  15. Go语言在大数据时代应用前景
  16. 写毕业论文更新目录时,如何让格式不会变?
  17. 关于使用jquery weui的picker部分遇到的问题及解决办法
  18. 华为运营商级路由器配置示例 | IPv4静态路由
  19. 蓝桥杯大学JAVA题型_蓝桥杯 2020年省赛真题 10月第二场 (Java 大学B组)
  20. 如何使用python实现简单爬取网页数据并导入MySQL中的数据库

热门文章

  1. php正则匹配sg-nc-wap_php正则匹配
  2. chrome 69 免安装_ElasticSearch安装elasticsearch-head插件
  3. 你的元器件为什么会无缘无故地失效了?
  4. FPGA学习之路—Vivado与Modelsim联合仿真
  5. 使用verilog设计实现QR分解
  6. linux 系统arp检测工具,linux网络常用诊断工具
  7. oracle无法修改字段类型,Oracle如何修改字段类型呢? 爱问知识人
  8. 20220130---CTF WEB方向刷题WP-----网页初始index.php/robots.txt
  9. java用if判断输入字符_java怎么用if判断输入的是不是数字
  10. P4016 负载平衡问题(最小费用最大流)