【EventLoop】从理解到精通事件循环

前言

什么是事件循环/EventLoop?

EventLoop JS 代码的执行分为同步任务代码异步任务代码当碰到同步代码时,直接在执行栈中执行,当碰到异步任务并且时机符合时(例如定时器的时间到了),就把异步代码添加到任务队列中当执行栈中的同步代码执行完毕后,才去任务队列中把异步代码拿到执行栈中执行其实这种返回轮询任务队列的过程就是 Event Loop;

什么是单线程?

JS 是单线程的,也就是说执行 JS 代码的线程只有一个( JS 引擎线程),我们也叫主线程,那为什么浏览器可以同时执行异步任务呢,因为浏览器的渲染进程是多线程的,当需要执行异步任务的时候,浏览器会帮我们启动另外一个线程来执行该任务! 浏览器中还有定时器线程、HTTP请求线程等,这些任务不是用来执行 JS 代码的,主要用来执行其他的一些任务!

比如主线程(JS 引擎线程)中碰到AJAX请求,就把这个任务交给线程HTTP请求线程去真正的发送请求,等到请求回来了之后,再将callback里需要执行的JS回调函数,交给JS引擎线程去执行,也就是说浏览器才是真正执行发送请求这个任务的角色,而JS只是负责执行最后的回调处理!

PS:`事件循环是js代码单线程语言执行机制,编译器是按照事件循环的机制来执行的; (1)先解析默认script宏任务,

(2)判断代码是同步还是异步,

(3)如果是同步,立即执行,

(4)如果是异步,就放入异步队列,微任务放入微任务队列,宏任务放入宏任务队列;

(5)当前排队所有的同步都执行完毕,开始执行异步,从微队列开始依次执行。

小试牛刀(练习一)

源代码如下:

javascript 复制代码
// 练习一 : 小试牛刀
setTimeout(function() {
  console.log(1)
}, 0);

new Promise(function executor(resolve) {
  console.log(2);
  for( var i=0 ; i<10000 ; i++ ) {
    i == 9999 && resolve();
  }
  console.log(3);
}).then(function() {
  console.log(4);
});
console.log(5);

分析打印结果如下:

javascript 复制代码
// 练习一 : 小试牛刀
setTimeout(function() {
  // 异步(宏任务 #1)
  console.log(1)
}, 0);

new Promise(function executor(resolve) {
  // 同步任务 #1
  console.log(2);
  for( var i=0 ; i<10000 ; i++ ) {
    i == 9999 && resolve();
  }
  // 同步任务 #2
  console.log(3);
}).then(function() {
  // 异步(微任务 #1)
  console.log(4);
});
  // 同步任务 #3
console.log(5);

// 依次打印结果:  2  3  5  4  1

PS: return Promise.resolve(undefined).then(); // 默认会返回这么一句话

渐入佳境(练习二)

源代码如下:

javascript 复制代码
// 练习二: 渐入佳境
setTimeout(() => {
  new Promise(resolve => {
      resolve();
  }).then(() => {
      console.log('test');
  });
    console.log(4);
}); 

new Promise(resolve => {
  resolve();
  console.log(1)
}).then(() => {
  console.log(3);
  Promise.resolve().then(() => {
    console.log('before timeout');
  // return Promise.resolve(undefined); // 默认会返回这么一句话
}).then(() => {
  Promise.resolve().then(() => {
    console.log('also before timeout')
    })
  })
})
console.log(2);

分析结果如下:

javascript 复制代码
  // 练习二: 渐入佳境
setTimeout(() => {
  // 宏 #1
  new Promise(resolve => {
      resolve();
  }).then(() => {
    // 微 #5
      console.log('test');
  });
  // 同 #3
    console.log(4);
}); 

new Promise(resolve => {
  // 同 #1
  resolve();
  // 微 #1
  console.log(1)
}).then(() => {
  // 微 #2
  console.log(3);
  Promise.resolve().then(() => {
  // 微 #3
    console.log('before timeout');
  // return Promise.resolve(undefined); // 默认会返回这么一句话
}).then(() => {
  Promise.resolve().then(() => {  
    // 微 #4
    console.log('also before timeout')
    })
  })
})
// 同 #2
console.log(2);

// 依次打印结果:  1 2 3 before timeout also before timeout  4  test

游刃有余(练习三)

源代码如下:

scss 复制代码
// 练习三: 游刃有余
setTimeout(function() {
    console.log(0);
});

new Promise((resolve, reject) => {
    console.log(1);
    resolve();
}).then(() => {
    console.log(2);
    new Promise((resolve, reject) => {    
        console.log(3);
        resolve();
    }).then(() => {
        console.log(4);
    }).then(() => {
        console.log(5);
    });
}).then(() => {
    console.log(6);
});

new Promise((resolve, reject) => {
    console.log(7);
    resolve();
}).then(() => {
    console.log(8);
});

分析结果如下:

scss 复制代码
// 练习三: 游刃有余
setTimeout(function() {
    // 宏 #1
    console.log(0);
});

new Promise((resolve, reject) => {
  // 同 #1
    console.log(1);
    resolve();
}).then(() => {
  // 微 #1
    console.log(2);
    new Promise((resolve, reject) => {    
        console.log(3);
        resolve();
    }).then(() => {
      // 微 #3
        console.log(4);
    }).then(() => {
      // 微 #5
        console.log(5);
    });
}).then(() => {
  // 微 #4
    console.log(6);
});

new Promise((resolve, reject) => {
  // 同 #2
    console.log(7);
    resolve();
}).then(() => {
  // 微 #2
    console.log(8);
});

结尾

在日益竞争的软件开发领域,掌握异步编程和事件循环的知识将使您在开发高性能、高可用性的应用程序上更加游刃有余。不断学习和实践是精通这些技术的关键,希望您能在实际项目中尝试应用这些概念,从而不断完善自己的技术水平。 逐字稿: 事件循环是js代码单线程语言执行机制,编译器是按照事件循环的机制来执行的;

(1)先解析默认script宏任务,

(2)判断代码是同步还是异步,

(3)如果是同步,立即执行,

(4)如果是异步,就放入异步队列,微任务放入微任务队列,宏任务放入宏任务队列;

(5)当前排队所有的同步都执行完毕,开始执行异步,从微队列开始依次执行。

相关推荐
明远湖之鱼12 分钟前
浅入理解跨端渲染:从零实现 React DSL 跨端渲染机制
前端·react native·react.js
悟忧1 小时前
规避ProseMirror React渲染差异带来的BUG
前端
小皮虾1 小时前
小程序云开发有类似 uniCloud 云对象的方案吗?有的兄弟,有的!
前端·javascript·小程序·云开发
Android疑难杂症1 小时前
鸿蒙Notification Kit通知服务开发快速指南
android·前端·harmonyos
T___T1 小时前
全方位解释 JavaScript 执行机制(从底层到实战)
前端·面试
阳懿1 小时前
meta-llama-3-8B下载失败解决。
前端·javascript·html
Qinana1 小时前
🌊 深入理解 CSS:从选择器到层叠的艺术
前端·css·程序员
IT_陈寒1 小时前
Python 3.12新特性实测:10个让你的代码提速30%的隐藏技巧 🚀
前端·人工智能·后端
史林枫1 小时前
JavaScript 中call和apply的详细讲解 —— 连10岁的小朋友都能看懂!
javascript·apply·call
闲人编程1 小时前
从零开发一个简单的Web爬虫(使用Requests和BeautifulSoup)
前端·爬虫·beautifulsoup·bs4·web·request·codecapsule