博客
关于我
JavaScript之运行机制,以及Event-Loop的讲解
阅读量:194 次
发布时间:2019-02-28

本文共 2184 字,大约阅读时间需要 7 分钟。

JavaScript事件循环机制详解

引入题目

在学习JavaScript时,很多开发者会遇到一个难以理解的问题:代码中某些操作会导致输出顺序与预期不符。例如,以下代码:

console.log(1); setTimeout(() => {    console.log(2)}, 0); console.log(3);

打印的顺序是1、3、2,而不是我们直觉上认为的1、2、3。这一现象的背后是JavaScript的单线程执行机制和事件循环机制的深层原理。接下来,我们将从基础到高级深入探讨JavaScript的运行机制。

单线程与JavaScript的运行机制

单线程的定义

单线程指的是在同一时间内,只能执行一项任务。JavaScript选择单线程作为其主要执行模型的原因在于其主要用于与用户交互和操作DOM(文档对象模型)。如果采用多线程,会引入复杂的同步问题,例如同时修改DOM时可能导致的数据不一致。

单线程的含义

单线程意味着所有任务必须排队等待前一个任务完成才能执行。这种机制虽然简单,但在处理大量异步任务时可能导致性能问题。因此,JavaScript引入了任务队列来处理异步任务。

任务队列的概念

任务队列是用来管理异步任务的容器。异步任务被分为两类:

  • 同步任务(Synchronous Task):直接在主线程上执行的任务,例如console.log()、DOM操作等。
  • 异步任务(Asynchronous Task):不直接在主线程上执行的任务,例如setTimeout()、网络请求等,这些任务会被放入任务队列中,等待主线程空闲时执行。
  • 任务队列的执行顺序

    任务队列中的任务按照先进先出的原则执行。当主线程完成所有同步任务后,会去任务队列中取出最早的异步任务进行执行。

    Event Loop的工作原理

    Event Loop是JavaScript执行异步任务的核心机制。它的执行过程可以分为以下几个阶段:

  • 执行当前宏任务(Macro Task):主线程执行当前脚本中的同步代码。
  • 处理微任务(Micro Task):在宏任务执行完成后,立即处理任务队列中的微任务。
  • 重复上述步骤直到所有任务完成:直到所有宏任务和微任务都被处理完毕,Event Loop才会终止。
  • 宏任务与微任务

    宏任务包括:

    • 主脚本(Script)的执行代码
    • setTimeout()
    • setInterval()
    • setImmediate()
    • I/O操作
    • UI渲染

    微任务包括:

    • process.nextTick()
    • Promise
    • MutationObserver

    宏任务和微任务的执行顺序

  • 执行当前宏任务。
  • 处理任务队列中的所有微任务。
  • 进入下一个宏任务循环。
  • 宏任务与微任务的执行顺序示例

    以下代码:

    setTimeout(() => {    console.log('a');}, 0);Promise.resolve().then(() => {    console.log('b');}).then(() => {    Promise.resolve('c').then((data) => {        setTimeout(() => {            console.log('d');        }, 0);        console.log('e');        return data;    }).then((data) => {        console.log(data);    });new Promise((resolve) => {    console.log('f');    resolve('g');}).then((res) => {    console.log(res);});console.log('h');

    将打印出:f, h, b, g, e, c, a, d。这是因为:

  • console.log('h')是主线程中的同步任务,立即执行。
  • fPromise中的同步代码,立即执行。
  • bg是同一个then链中的微任务,按顺序执行。
  • ecthen链中的同步操作。
  • adsetTimeout触发,按顺序执行。
  • 题目2解析

    以下代码:

    console.log(1);let a = setTimeout(() => {    console.log(2);}, 0);console.log(3);Promise.resolve(4).then(() => {    console.log(b);    clearTimeout(a);});console.log(5);

    打印结果为:1, 3, 5, 4。原因在于:

  • console.log(1)console.log(3)是同步任务,按顺序执行。
  • setTimeout生成的a被立即放入任务队列。
  • Promise.resolve(4)生成一个新的then,在clearTimeout(a)之前执行。
  • clearTimeout(a)确保在a执行完成后取消定时任务。
  • 通过以上分析,我们可以清晰地理解JavaScript事件循环机制的运行原理及其在实际开发中的应用。掌握这些知识对于优化和调试JavaScript代码至关重要。

    转载地址:http://wgni.baihongyu.com/

    你可能感兴趣的文章
    OpenCV_ cv2.imshow()
    查看>>
    opencv_core.dir/objects.a(vs_version.rc.obj)‘ is incompatible with i386:x86-64 output
    查看>>
    opencv——图像缩放1(resize)
    查看>>
    opencv——最简单的视频读取
    查看>>
    Opencv——模块介绍
    查看>>
    OpenCV与AI深度学习 | 2024年AI初学者需要掌握的热门技能有哪些?
    查看>>
    OpenCV与AI深度学习 | CIB-SE-YOLOv8: 优化的YOLOv8, 用于施工现场的安全设备实时检测 !
    查看>>
    OpenCV与AI深度学习 | CoTracker3:用于卓越点跟踪的最新 AI 模型
    查看>>
    OpenCV与AI深度学习 | OpenCV中八种不同的目标追踪算法
    查看>>
    OpenCV与AI深度学习 | OpenCV图像拼接--Stitching detailed使用与参数介绍
    查看>>
    OpenCV与AI深度学习 | OpenCV如何读取仪表中的指针刻度
    查看>>
    OpenCV与AI深度学习 | OpenCV常用图像拼接方法(一) :直接拼接
    查看>>
    OpenCV与AI深度学习 | OpenCV常用图像拼接方法(三):基于特征匹配拼接
    查看>>
    OpenCV与AI深度学习 | OpenCV常用图像拼接方法(二) :基于模板匹配拼接
    查看>>
    OpenCV与AI深度学习 | OpenCV常用图像拼接方法(四):基于Stitcher类拼接
    查看>>
    OpenCV与AI深度学习 | OpenCV快速傅里叶变换(FFT)用于图像和视频流的模糊检测(建议收藏!)
    查看>>
    OpenCV与AI深度学习 | PaddleOCR 2.9 发布, 正式开源文本图像智能分析利器
    查看>>
    OpenCV与AI深度学习 | SAM2(Segment Anything Model 2)新一代分割一切大模型介绍与使用(步骤 + 代码)
    查看>>
    OpenCV与AI深度学习 | T-Rex Label !超震撼 AI 自动标注工具,开箱即用、检测一切
    查看>>
    OpenCV与AI深度学习 | YOLO11介绍及五大任务推理演示(目标检测,图像分割,图像分类,姿态检测,带方向目标检测)
    查看>>