前置知识点:

事件循环机制

相信读者读完以上推荐的文章后,已经知道事件循环机制是怎么一回事了吧,也能从容应对面试。接下来我要谈谈自己的理解:

为什么会有事件循环机制

  • js设计之初就是单线程模式,代码也都是顺序执行,当遇到因为大量计算、http请求等需要额外的等待时间时,浏览器用户就会体验到卡顿了,所以所有的设计和改进初衷只有一个就是要快

事件循环机制的产生

  • 浏览器说我的内核是多线程,可以辅助JS引擎线程啊,Web Worker线程提供大量计算辅助(不能操作DOM),事件触发线程定时触发器线程异步http请求线程


  • 执行栈(先进后出),由JS引擎线程控制,引用下面这个例子谈谈自己的理解:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    console.log('script start');
    
    setTimeout(function() {
    console.log('setTimeout');
    }, 0); Promise.resolve().then(function() {
    console.log('promise1');
    }).then(function() {
    console.log('promise2');
    });
    console.log('script end'); // "script end"
    // "promise1"
    // "promise2"
    // "setTimeout"
  • ES5还没有Promise时代,异步回调很常见,上面例子中,通过解读Promise源码(前端面试必考题Promise的源码解析),我们可以把Promise转换成如下图式回调(个人理解,文章中的Promise源码也只是模拟,大部分浏览器已经原生支持)。

    1. 打印完script start, script end主执行栈出栈,如果Promise.resolve().then换成new Promise(executor),脑补Promise换成回调函数,那么这个函数一执行,executor函数也就执行了,然后遇到异步回调,回调函数被其它对应的线程接手,启动观察者模式,完成后回调函数被推入事件任务队列,等待执行栈空了进入主线程执行

    2. 以上这种在异步函数中放同步函数的例子,为了合理解释输出顺序而推出了microtasks微任务的概念,请看下面的例子,脑补Promise换成回调函数,Promise.prototype.then内部执行了return new Promise(),js引擎在捕捉到Promise时,放到了由js引擎自身控制的微任务队列等待执行,也就造成promise1、2、3、4错开打印

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      console.log('script start');
      
      new Promise(function(resolve, reject) {
      console.log('promise1');
      resolve();
      }).then(function() {
      console.log('promise2');
      }); new Promise(function(resolve, reject) {
      console.log('promise3');
      resolve();
      }).then(function() {
      console.log('promise4');
      }); console.log('script end'); // "promise1"
      // "promise3"
      // "script end"
      // "promise2"
      // "promise4"
    3. micro 大专栏  对javascript EventLoop事件循环机制不一样的理解tasks微任务的概念完全为了解释异步函数中放同步函数的场景,而且各类文章和面试都是这种题目和例子,在实际开发过程中,你会在Promise中这么写么?,在我看来这种比较打印顺序太过于理论,而且可能会混乱你的思绪。就像下面的例子,Promiseresolve决定了Promise状态,就像在回调函数中满足了条件才会继续执行,例子中只是用setTimeout模拟异步请求,用之前的理论你可能觉得setTimeout被放入了事件任务队列,那没有resolvePromise怎么解释呢?(放到微任务里一直阻碍第一个setTimeout宏任务执行吗?显然是不可能的,这不是跟设计事件循环机制初衷冲突了么

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      setTimeout(function() {
      console.log('setTimeout');
      }, 0); new Promise(function(resolve, reject) {
      setTimeout(function() { // 模拟异步请求
      console.log('promise1');
      resolve();
      }, 0);
      }).then(function() {
      console.log('promise2');
      }); new Promise(function(resolve, reject) {
      // resolve(); 注释掉resolve,使Promise一直处于‘pending’状态
      }).then(function() {
      console.log('promise2');
      }); // "setTimeout"
      // "promise1"
      // "promise2"
    4. 个人认为把Promiseasync/await脑补成原始的回调函数(模拟源码中模拟异步是用的setTimeout函数),而js引擎捕捉到setTimeout, setInterval就转给定时触发器线程处理,捕捉到XMLHttpReuqest, fetch就转给异步http请求线程,跟事件触发线程一起管理着事件任务队列,微任务的概念可以看作是当事件触发线程遇到几乎同时需要把回调函数放到事件任务队列时,Promise内部的异步标识函数优先级高于setTimeout函数吧,以上例子中没有执行resolvePromise状态一直处于’pending’,事件触发线程压根没有放入到事件任务队列,总之浏览器会安排的妥妥的,不要打架,虽然js引擎线程只有一个(听我指挥排好队,咱们这都是同步代码执行ms级别,我开了很多其它线程处理需要等待的代码了)。以下例子模拟所谓的几乎同时把回调函数放到事件任务队列,记得把Promise脑补成原始的回调函数。仿佛回到了没有微任务的时代。

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      setTimeout(function() {
      console.log('setTimeout');
      }, 0); new Promise(function(resolve, reject) {
      setTimeout(function() { // 模拟异步请求
      console.log('promise1');
      resolve();
      }, 2000);
      }).then(function() {
      console.log('promise2');
      }); new Promise(function(resolve, reject) {
      setTimeout(function() { // 模拟异步请求
      console.log('promise3');
      resolve();
      }, 2000);
      }).then(function() {
      console.log('promise4');
      }); // "setTimeout"
      // "promise1"
      // "promise2"
      // "promise3"
      // "promise4"

以上内容纯属未深入了解js情况下的个人理解,感觉是在努力摒弃微任务的概念,回归ES5回调函数时代,便于自身理解事件循环机制而做出的遐想。

多环境下的事件循环机制

在node环境、浏览器环境以及各个不同版本下js引擎处理的方式还不太一样。node比浏览器还复杂些

对javascript EventLoop事件循环机制不一样的理解的更多相关文章

  1. JavaScript的事件循环机制浅析

    前言 JavaScript是一门单线程的弱类型语言,但是我们在开发中,经常会遇到一些需要异步或者等待的处理操作. 类似ajax,亦或者ES6中新增的promise操作用于处理一些回调函数等. 概念 在 ...

  2. 【运行机制】 JavaScript的事件循环机制总结 eventLoop

    0.从个例子开始 //code-01 console.log(1) setTimeout(() => { console.log(2); }); console.log(3); 稍微有点前端经验 ...

  3. javascript的事件循环机制

    JavaScript是一门编程语言,既然是编程语言那么就会有执行时的逻辑先后顺序,那么对于JavaScript来说这额顺序是怎样的呢? 首先我们我们需要明确一点,JavaScript是单线程语言.所谓 ...

  4. JavaScript 运行机制:Event事件循环机制

    JavaScript Event事件循环机制 JS是单线程的,浏览器只分配一个主线程给JS.一次只能执行一个任务,当前任务执行完后在可以执行下一个任务.任务多时,就会形成任务队列排队等待执行.但是非常 ...

  5. 深入理解JavaScript事件循环机制

    前言 众所周知,JavaScript 是一门单线程语言,虽然在 html5 中提出了 Web-Worker ,但这并未改变 JavaScript 是单线程这一核心.可看HTML规范中的这段话: To ...

  6. javascript事件循环机制 浅尝手记

    引入 众所周知Javascript是一个单线程的机制,虽然可以依托多线程的浏览器实现页面如何实现页面复杂的渲染.事件响应,但仍不会改变其单线程的本质:所以对于js的事件循环机制的了解是一个前端人员的必 ...

  7. selector.select(500); EventLoop及事件循环机制 netty 在半透明做代理网关下 对请求的批处理

    Netty框架学习之路(五)—— EventLoop及事件循环机制 - 懋懋之为 - CSDN博客 https://blog.csdn.net/tjreal/article/details/79751 ...

  8. 深入浅出Javascript事件循环机制

    一.JS单线程.异步.同步概念 众所周知,JS是单线程(如果一个线程删DOM,一个线程增DOM,浏览器傻逼了-所以只能单着了),虽然有webworker酱紫的多线程出现,但也是在主线程的控制下.web ...

  9. 浏览器中的JavaScript事件循环机制

    浏览器的事件循环机制是HTML中定义的规范. JavaScript有一个主线程和调用栈,所有的任务都会被放到调用栈等待主线程执行. JS调用栈 是一种先进后出的数据结构.当函数被调用时,会被添加到栈中 ...

随机推荐

  1. 吴裕雄--天生自然运维技术:LMT

    LMT,Local Maintenance Terminal的缩写,意思是本地维护终端.LMT是一个逻辑概念.LMT连接到RNC外网,提供NODE B操作维护的用户界面. LMT也是许可证管理技术Li ...

  2. 吴裕雄--天生自然 pythonTensorFlow自然语言处理:文本数据预处理--生成训练文件

    import sys import codecs # 1. 参数设置 MODE = "PTB_TRAIN" # 将MODE设置为"PTB_TRAIN", &qu ...

  3. push 空内容push入数组会占位

    #!/usr/bin/perl use strict; use warnings; ==)?:'';my @arr; ==)?:''; '; my $line = join "|" ...

  4. Git log 中文乱码

    以下三条命令搞定(系统是centos  7.4) git config --global i18n.commitencoding utf-8 git config --global i18n.logo ...

  5. 40)PHP,mysql_fetch_row和mysql_fetch_array和mysql_fetch_assoc的区别

    分析: mysql_fetch_row,这个函数是从结果集中取一行作为枚举数据,从和指定的结果标识关联的结果集中取得一行数据并作为数组返回.每个结果的列储存在一个数组的单元中,偏移量从 开始. 注意, ...

  6. python学习笔记(15)pymysql数据库操作

    pymysql数据库操作 1.什么是PyMySQL 为了使python连接上数据库,你需要一个驱动,这个驱动是用于与数据库交互的库. PyMySQL : 这是一个使Python连接到MySQL的库,它 ...

  7. sqlserver命令创建数据库和表 demo

    由于sqlserver用起来很不爽 可以尝试用vscode+sqlserver插件玩玩 友情提示 在vscode中新建一个.sql 并配置好与sqlserver的连接 利用sql会有提示创建表 数据库 ...

  8. 如何修改tomcat名称

    修改tomcat名称为  Tomcat-jx-1

  9. 吴裕雄--天生自然python学习笔记:pandas模块强大的数据处理套件

    用 Python 进行数据分析处理,其中最炫酷的就属 Pa ndas 套件了 . 比如,如果我 们通过 Requests 及 Beautifulsoup 来抓取网页中的表格数据 , 需要进行较复 杂的 ...

  10. RS232串口通信详解

    串口是计算机上一种非常通用的设备通信协议. ---------------------------------串口的引脚定义: 9芯 信号方向来自 缩写 描述 1 调制解调器 CD 载波检测 2 调制 ...