Maison >interface Web >js tutoriel >Plongée profonde dans l'asynchronie : microtâches, macrotâches et boucle d'événements

Plongée profonde dans l'asynchronie : microtâches, macrotâches et boucle d'événements

Linda Hamilton
Linda Hamiltonoriginal
2025-01-01 08:23:10995parcourir

Deep Dive in Asynchrony: Microtasks, Macrotasks, and the Event Loop

La nature asynchrone de JavaScript peut sembler magique jusqu'à ce que vous plongez dans la mécanique. La sauce secrète réside dans sa boucle d'événements, qui orchestre deux acteurs clés : les microtâches et les macrotâches. Mais que sont-ils, comment fonctionnent-ils et pourquoi sont-ils importants ? Perçons le mystère avec une analyse approfondie, des exemples et des conseils pour maîtriser ce concept.


La boucle d'événements et les files d'attente de tâches

Le moteur JavaScript exécute le code dans un seul thread. Pour gérer les opérations asynchrones, il s'appuie sur la boucle d'événements, qui assure la coordination entre la pile d'appels et les files d'attente de tâches. Ces files d'attente de tâches sont divisées en deux catégories : les microtâches et les macrotâches.

Microtâches

Les microtâches sont des tâches hautement prioritaires qui doivent être exécutées dès que le code JavaScript en cours d'exécution est terminé et que la pile d'appels est vide. Ils garantissent des actions de suivi rapides et des états cohérents. Les exemples courants incluent :

  • Promesses (.then, async/await)
  • MutationObserver rappels

Macrotâches

Les macrotâches sont des tâches de moindre priorité que la boucle d'événements ne gère qu'une fois que toutes les microtâches ont été exécutées. Ils gèrent des opérations différées plus importantes et des événements externes. Les exemples courants incluent :

  • Minuteries (setTimeout, setInterval)
  • MessageChannel rappels
  • Opérations d'E/S

Il existe également requestAnimationFrame, qui ne fait partie d'aucune des deux files d'attente. Il se synchronise avec le cycle de rendu du navigateur, ce qui le rend idéal pour des animations fluides.


Comment ça marche

Voici comment la boucle d'événements traite les tâches :

  1. Exécute d'abord le code synchrone.
  2. Efface la file d'attente des microtâches.
  3. Exécute une tâche de la file d'attente des macrotâches.
  4. Répète les étapes 2 et 3 jusqu'à ce que toutes les tâches soient traitées.

Cette hiérarchisation garantit que les tâches hautement prioritaires telles que les promesses sont résolues avant les opérations moins urgentes comme les minuteries.


Un exemple en action

Vous trouverez ci-dessous un extrait de code pratique pour illustrer l'interaction entre le code synchrone, les microtâches, les macrotâches et requestAnimationFrame :

console.log('Synchronous code starts');

// Macrotask: setTimeout
setTimeout(() => {
  console.log('Macrotask: setTimeout');
}, 0);

// Macrotask: setInterval
const intervalId = setInterval(() => {
  console.log('Macrotask: setInterval');
  clearInterval(intervalId);
}, 100);

// Microtask: Promise
Promise.resolve().then(() => {
  console.log('Microtask: Promise then 1');
  Promise.resolve().then(() => {
    console.log('Microtask: Promise then 2');
  });
});

// Microtask: MutationObserver
const observer = new MutationObserver(() => {
  console.log('Microtask: MutationObserver');
});
const targetNode = document.createElement('div');
observer.observe(targetNode, { attributes: true });
targetNode.setAttribute('data-test', 'true');

// Macrotask: MessageChannel
const channel = new MessageChannel();
channel.port1.onmessage = () => {
  console.log('Macrotask: MessageChannel');
};
channel.port2.postMessage('Test');

// requestAnimationFrame
requestAnimationFrame(() => {
  console.log('Outside task queues: requestAnimationFrame');
});

console.log('Synchronous code ends');

Résultat attendu

La séquence de sortie permet de clarifier la priorisation :

  1. Code synchrone : "Le code synchrone démarre", "Le code synchrone se termine"
  2. Microtâches :
    • « Microtâche : Promesse puis 1 »
    • « Microtâche : Promesse puis 2 »
    • « Microtâche : MutationObserver »
  3. Macrotâches :
    • « Macrotâche : setTimeout »
    • « Macrotâche : MessageChannel »
    • « Macrotâche : setInterval »
  4. requestAnimationFrame : « En dehors des files d'attente de tâches : requestAnimationFrame »

Exploration approfondie : microtâches et macrotâches

Microtâches : caractéristiques clés

  • Exécuter immédiatement une fois le code synchrone terminé.
  • Idéal pour les petites mises à jour hautement prioritaires comme la résolution de promesses ou la réaction aux mutations du DOM.
  • Exemples : Promesses, MutationObserver.

Macrotâches : caractéristiques clés

  • Exécuter uniquement une fois que toutes les microtâches ont été effacées.
  • Utilisé pour des opérations plus importantes et moins prioritaires ou pour la gestion d'événements externes.
  • Exemples : minuteries, MessageChannel.

requestAnimationFrame : L'intrus sorti

Bien qu'il ne fasse pas partie des files d'attente de tâches, requestAnimationFrame joue un rôle unique dans l'asynchronie. Il planifie l'exécution du code avant la prochaine repeinture du navigateur, garantissant ainsi un minimum de pertes d'images et des animations plus fluides.


Conclusion

L'interaction entre les microtâches, les macrotâches et la boucle d'événements est au cœur de l'asynchronie de JavaScript. En comprenant et en exploitant ces concepts, vous pouvez écrire un code plus efficace, plus maintenable et plus performant. N'oubliez pas : les microtâches en premier, les macrotâches en second et requestAnimationFrame pour le peaufinage visuel. Bon codage !

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!

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn