


Un article pour parler du mécanisme de boucle d'événement dans Node.js
Cet article vous fera comprendre le mécanisme de boucle d'événement (boucle temporelle) dans Node.js. J'espère qu'il vous sera utile !
Aujourd'hui, nous allons découvrir la boucle d'événements dans nodeJs. La compréhension de la boucle événementielle a toujours été une grande difficulté pour moi. J'espère surmonter cette difficulté grâce à cette étude. J'espère également approfondir ma compréhension et mon impression de la boucle événementielle à travers ce blog.
libuv
Avant d'apprendre la boucle d'événements, comprenez d'abord la libuv du nœud. libuv est responsable de l'implémentation de différents modèles d'E/S sur différents systèmes d'exploitation et résume différentes implémentations dans des API pouvant être appliquées à des applications tierces.
Question
Avant d'apprendre formellement la boucle d'événements, réfléchissons à une question
setTimeout(() => { console.log("timer1"); Promise.resolve().then(() => { console.log("promise1"); }); }, 0); setTimeout(() => { console.log("timer2"); Promise.resolve().then(() => { console.log("promise2"); }); }, 0);
Quel est le résultat de l'exécution de ce code dans le navigateur ?
Quel est le résultat de son exécution dans node ?
Avant node8.6 :
Après node8.6 :
Pourquoi il y a un tel résultat, nous l'analyserons plus tard !
Tout d'abord, regardons une image :
Vous pouvez voir 6 étapes dans l'image, à savoir : minuteries, rappels en attente, inactivité/préparation, sondage, vérification, clôture des rappels.phase des minuteries : exécute principalement les rappels setTimeOut, setInterval
phase de rappels en attente : exécute certaines erreurs d'appel système, telles que les rappels d'erreurs de communication réseau
phase d'inactivité/préparation : utilisée uniquement dans le système (nous avons aucun contrôle ni interférence à ce stade)
étape d'interrogation : obtenez de nouveaux événements d'E/S, comme l'obtention d'un rappel d'E/S pour la lecture d'un fichier.
Dans des circonstances appropriées, nodejs bloquera dans cette phasephase de vérification : exécutez le rappel setImmediate
Par exemple, exécutez la destory de sokect, fermez le rappel d'événement
Chaque phase suit un
FIFO (
) règles pour exécuter les tâches dans la file d'attente des tâches. Parmi ces six étapes, nous devons nous concentrer sur les étapes timers, poll, check. La plupart des tâches asynchrones de notre développement quotidien sont traitées au cours de ces trois étapes. timers
Parlons d'abord de l'étape des timers. timers est la première étape de la boucle d'événements, nodejs vérifiera s'il y a un timer expiré et, si c'est le cas, mettra son rappel dans la file d'attente. Cependant, nodejs ne peut pas garantir que le timer exécutera le rappel immédiatement lorsque l'événement prédéfini arrive. En effet, la vérification de l'expiration du timer par nodejs n'est pas nécessairement fiable. Elle sera affectée par d'autres programmes en cours d'exécution sur la machine, ou elle le sera. Rencontrez le fil principal actuel. Pas de situation inactive.
Pour l'incertitude ici, le site officiel donne un exemple :
Déclarez d'abord un setTimeOut, puis lisez un fichier en externe Lorsque l'opération de lecture du fichier dépasse le timer, l'opération de lecture du fichier définira le timer. est retardé. C'est la situation dans laquelle le thread principal n'est pas inactif comme mentionné précédemment.
poll
La phase d'interrogation effectue principalement deux choses : 1. Traitez la file d'attente des tâches de la phase d'interrogation
2 Lorsqu'un minuteur a expiré, exécutez sa fonction de rappel
. dans l'image ci-dessus, nous pouvons également voir :Après avoir exécuté les tâches de la file d'attente des tâches d'interrogation dans la phase d'interrogation, il vérifiera s'il existe un setImmediate prédéfini. S'il y en a, il entrera dans la phase de vérification. Sinon, nodejs. bloquera ici.
Ici, nous avons une question. S'il est bloqué lors de la phase de sondage, le minuteur que nous avons défini ne pourra-t-il pas être exécuté ? En faitLorsque la boucle d'événements est bloquée dans la phase d'interrogation, nodejs aura un mécanisme de vérification qui vérifiera si la file d'attente des minuteries est vide. Si elle n'est pas vide, elle entrera à nouveau dans la phase des minuteries.
check
La phase de vérification exécute principalement la fonction de rappel de setImmediate. 小总结
event-loop的每个阶段都有一个队列,当event-loop达到某个阶段之后,将执行这个阶段的任务队列,直到队列清空或者达到系统规定的最大回调限制之后,才会进入下一个阶段。当所有阶段都执行完成一次之后,称event-loop完成一个tick。
案例
上面我们说完了event-loop的理论部分,但是光有理论我们也还是不能很清晰的理解event-loop。下面我们就根据几个demo来更加深入的理解下event-loop!
demo1
const fs=require('fs')
fs.readFile('test.txt',()=>{
console.log('readFile')
setTimeout(()=>{
console.log('settimeout');
},0)
setImmediate(()=>{
console.log('setImmediate')
})
})
Copier après la connexion执行结果:

可见执行结果跟我们前面的分析时一致的!
demo2
const fs = require("fs");
const EventEmitter = require("events").EventEmitter;
let pos = 0;
const messenger = new EventEmitter();
messenger.on("message", function (msg) {
console.log(++pos + " message:" + msg); //
});
console.log(++pos + " first"); //
process.nextTick(function () {
console.log(++pos + " nextTick"); //
});
messenger.emit("message", "hello!");
fs.stat(__filename, function () {
console.log(++pos + " stat"); //
});
setTimeout(function () {
console.log(++pos + " quick timer"); //
}, 0);
setTimeout(function () {
console.log(++pos + " long timer"); //
}, 30);
setImmediate(function () {
console.log(++pos + " immediate"); //
});
console.log(++pos + " last"); //
Copier après la connexion结果:

了解下浏览器和node的event-loop差异在什么地方
在node 8.6 之前:
浏览器中的微任务队列会在每个宏任务执行完成之后执行,而node中的微任务会在事件循环的各个阶段之间执行,即每个阶段执行完成之后会去执行微任务队列。
在8.6之后:
浏览器和node中微任务的执行是一致的!
所以,在文章开头,我们提出的思考的问题就有了结果。
关于 process.nextTick()和setImmediate
process.nextTick()
语法:process.nextTick(callback,agrs)
执行时机:
这个函数其实是独立于 Event Loop 之外的,它有一个自己的队列,当每个阶段完成后,如果存在 nextTick 队列,就会清空队列中的所有回调函数,并且优先于其他 microtask 执行。递归的调用process.nextTick()
会导致I/O starving,官方推荐使用setImmediate()
关于starving现象的说明:
const fs = require("fs");
fs.readFile("test.txt", (err, msg) => {
console.log("readFile");
});
let index = 0;
function handler() {
if (index >= 30) return;
index++;
console.log("nextTick" + index);
process.nextTick(handler);
}
handler();
Copier après la connexion运行结果:

可以看到,等到nextTick函数呗执行30次之后,读取文件的回调才被执行!这样的现象被称为 I/O 饥饿。
当我们把 process.nextTick 换为 setImmediate
const fs = require("fs");
fs.readFile("test.txt", (err, msg) => {
console.log("readFile");
});
let index = 0;
function handler() {
if (index >= 30) return;
index++;
console.log("nextTick" + index);
setImmediate(handler);
}
handler();
Copier après la connexion结果:

造成这两种差异的原因是,嵌套调用的setImmediate的回调被排到了下一次event-loop中去!
event-loop核心思维导图

结束语
通过今天的学习,让我对event-loop的理解更深刻了。那么,下次见。好好学习,天天向上!

更多编程相关知识,请访问:编程视频!!
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!
const fs=require('fs') fs.readFile('test.txt',()=>{ console.log('readFile') setTimeout(()=>{ console.log('settimeout'); },0) setImmediate(()=>{ console.log('setImmediate') }) })

const fs = require("fs"); const EventEmitter = require("events").EventEmitter; let pos = 0; const messenger = new EventEmitter(); messenger.on("message", function (msg) { console.log(++pos + " message:" + msg); // }); console.log(++pos + " first"); // process.nextTick(function () { console.log(++pos + " nextTick"); // }); messenger.emit("message", "hello!"); fs.stat(__filename, function () { console.log(++pos + " stat"); // }); setTimeout(function () { console.log(++pos + " quick timer"); // }, 0); setTimeout(function () { console.log(++pos + " long timer"); // }, 30); setImmediate(function () { console.log(++pos + " immediate"); // }); console.log(++pos + " last"); //

process.nextTick(callback,agrs)
process.nextTick()
会导致I/O starving,官方推荐使用setImmediate()
const fs = require("fs"); fs.readFile("test.txt", (err, msg) => { console.log("readFile"); }); let index = 0; function handler() { if (index >= 30) return; index++; console.log("nextTick" + index); process.nextTick(handler); } handler();
const fs = require("fs"); fs.readFile("test.txt", (err, msg) => { console.log("readFile"); }); let index = 0; function handler() { if (index >= 30) return; index++; console.log("nextTick" + index); setImmediate(handler); } handler();

Outils d'IA chauds

Undresser.AI Undress
Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover
Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool
Images de déshabillage gratuites

Clothoff.io
Dissolvant de vêtements AI

AI Hentai Generator
Générez AI Hentai gratuitement.

Article chaud

Outils chauds

Bloc-notes++7.3.1
Éditeur de code facile à utiliser et gratuit

SublimeText3 version chinoise
Version chinoise, très simple à utiliser

Envoyer Studio 13.0.1
Puissant environnement de développement intégré PHP

Dreamweaver CS6
Outils de développement Web visuel

SublimeText3 version Mac
Logiciel d'édition de code au niveau de Dieu (SublimeText3)

Le service Node construit sur une base non bloquante et piloté par les événements présente l'avantage d'une faible consommation de mémoire et est très adapté à la gestion de requêtes réseau massives. Dans le contexte de demandes massives, les questions liées au « contrôle de la mémoire » doivent être prises en compte. 1. Le mécanisme de récupération de place du V8 et les limitations de mémoire Js sont contrôlés par la machine de récupération de place

Cet article vous donnera une compréhension approfondie de la mémoire et du garbage collector (GC) du moteur NodeJS V8. J'espère qu'il vous sera utile !

Le module de fichiers est une encapsulation des opérations de fichiers sous-jacentes, telles que l'ajout de lecture/écriture/ouverture/fermeture/suppression de fichiers, etc. La plus grande caractéristique du module de fichiers est que toutes les méthodes fournissent deux versions de **synchrone** et ** asynchrone**, with Les méthodes avec le suffixe sync sont toutes des méthodes de synchronisation, et celles qui n'en ont pas sont toutes des méthodes hétérogènes.

Le choix d'une image Docker pour Node peut sembler trivial, mais la taille et les vulnérabilités potentielles de l'image peuvent avoir un impact significatif sur votre processus CI/CD et votre sécurité. Alors, comment choisir la meilleure image Docker Node.js ?

Node 19 est officiellement publié. Cet article vous donnera une explication détaillée des 6 fonctionnalités majeures de Node.js 19. J'espère qu'il vous sera utile !

Comment Node.js fait-il le GC (garbage collection) ? L’article suivant vous guidera à travers cela.

La boucle d'événements est un élément fondamental de Node.js et permet une programmation asynchrone en garantissant que le thread principal n'est pas bloqué. Comprendre la boucle d'événements est crucial pour créer des applications efficaces. L'article suivant vous donnera une compréhension approfondie de la boucle d'événements dans Node. J'espère qu'il vous sera utile !

La raison pour laquelle le nœud ne peut pas utiliser la commande npm est que les variables d'environnement ne sont pas configurées correctement. La solution est la suivante : 1. Ouvrez "Propriétés système" ; 2. Recherchez "Variables d'environnement" -> "Variables système", puis modifiez l'environnement. variables ; 3. Recherchez l'emplacement du dossier nodejs ; 4. Cliquez sur "OK".
