JS运行机制

1. 单线程

JavaScript在同一时间内只能做一件事,即JS单线程。也常被称为“阻塞式执行”。

2. 任务队列

‘任务队列’指一个先进先出的队列,它里面存着各种事件和任务。

任务分为两种:同步任务和异步任务
区别同步和异步:有无阻塞。

3. 同步任务

指在任务队列中排队执行的任务,需要等前一个任务执行完才可以执行。

4. 异步任务

异步任务则不会阻塞。

常见异步任务:

  • setTimeout和setInterval
  • DOM事件
  • Promise
  • process.nextTick
  • fs.readFile
  • http.get
  • 异步函数:如果在函数返回的时候,调用者还不能够得到预期结果,而是需要在将来通过一定的手段得到,那么这个函数就是异步的。

同步任务和异步任务分别进入不同的场所:

  1. 同步任务进入主线程,异步任务进入Event Table并注册回调函数
  2. 当函数注册完毕后会将这个函数移入Event Queue
  3. 主线程内的任务执行完毕后会取Event Queue读取对应函数,进入主线程执行

不断重复步骤,形成了事件循环(Event Loop)

如下图:
在这里插入图片描述
JS异步机制:遇到宏任务先执行宏任务,将宏任务放入宏任务 的Queue,然后再执行微任务,将微任务放入微任务的 Queue
(即先遇到宏任务先执行宏任务,然后执行所有的微任务,再执行宏任务)

如下图:
在这里插入图片描述

5. 宏任务

常见宏任务:

  • I/O
  • setTimeout
  • setInterval
  • setImmdiate
  • requestAnimationFrame
6. 微任务
  • process.nextTick
  • Promise
  • Promise.then
  • MutationObserver

例子1:

setTimeout(() => {
    //执行后 回调一个宏事件
    console.log('内层宏事件3')
}, 0)
console.log('外层宏事件1');

new Promise((resolve) => {
    console.log('外层宏事件2');
    resolve()
}).then(() => {
    console.log('微事件1');
}).then(()=>{
    console.log('微事件2')
})

外层宏事件1
外层宏事件2
微事件1
微事件2
内层宏事件3
  • 首先浏览器执行js进入第一个宏任务进入主线程, 遇到 setTimeout分发到宏任务Event Queue

  • 遇到 console.log()直接执行 输出 外层宏事件1

  • 遇到 Promisenew Promise直接执行 输出 外层宏事件2

  • 执行then被分发到微任务Event Queue

  • 第一轮宏任务执行结束,开始执行微任务 打印 微事件1. 微事件2

  • 第一轮微任务执行完毕,执行第二轮宏事件,打印setTimeout里面内容内层宏事件3

例子2:

//主线程直接执行
console.log('1');
//丢到宏事件队列中
setTimeout(function() {
    console.log('2');
    process.nextTick(function() {
        console.log('3');
    })
    new Promise(function(resolve) {
        console.log('4');
        resolve();
    }).then(function() {
        console.log('5')
    })
})
//微事件1
process.nextTick(function() {
    console.log('6');
})
//主线程直接执行
new Promise(function(resolve) {
    console.log('7');
    resolve();
}).then(function() {
    //微事件2
    console.log('8')
})
//丢到宏事件队列中
setTimeout(function() {
    console.log('9');
    process.nextTick(function() {
        console.log('10');
    })
    new Promise(function(resolve) {
        console.log('11');
        resolve();
    }).then(function() {
        console.log('12')
    })
})

  • 首先浏览器执行js进入第一个宏任务进入主线程, 直接打印console.log('1')

  • 遇到 setTimeout分发到宏任务Event Queue

  • 遇到 process.nextTick丢到微任务Event Queue

  • 遇到Promisenew Promise直接执行 输出 console.log('7');

  • 执行then被分发到微任务Event Queue

  • 第一轮宏任务执行结束,开始执行微任务 打印 6,8

  • 第一轮微任务执行完毕,执行第二轮宏事件,执行setTimeout

  • 先执行主线程宏任务,在执行微任务,打印2,4,3,5

  • 在执行第二个setTimeout,同理打印9,11,10,12

  • 整段代码,共进行了三次事件循环,完整的输出为1,7,6,8,2,4,3,5,9,11,10,12。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值