用几道面试题来看JavaScript执行机制

前面的话

根据javascript的运行环境,锁定它为单线程,任务需要排队执行,如果网站资源比较大,这样会导致浏览器加载会很慢,但实际上并没有,大家肯定立刻想到了同步和异步。

所谓的同步和异步也是在排队,只是排队的地方不同。

同步和异步

同步任务进入主线程排队,异步任务进入事件队列中排队

同步任务和异步任务进入到不同的队列中,也就是上面讲的在不同地方排队。

同步任务进入主线程,异步任务进入事件队列,主线程任务执行完毕,事件队列中有等待执行的任务进入主线程执行,直到事件队列中任务全部执行完毕。

开胃菜

console.log('a')

setTimeout(function(){
    console.log('b')
}, 200)

setTimeout(function(){
    console.log('c')
}, 0)

console.log('d')

a d c b

从上到下,该进入主线程的进入主线程,该进入事件队列的进入事件队列。

那么主线程中存在console.log('a')和console.log('d'),定时器setTimeout延迟一段时间执行,顾名思义异步任务进入事件队列中,等待主线程任务执行完毕,再进入主线程执行。

定时器的延迟时间为0并不是立刻执行,只是代表相比于其他定时器更早的进入主线程中执行。

加一盘

for(var i = 0; i < 10; i++) {
    setTimeout(function() {
        console.log(i)
    }, 1000)
}

结果:十个10

每次for循环遇到setTimeout将其放入事件队列中等待执行,直到全部循环结束,i作为全局变量当循环结束后i = 10,再来执行setTimeout时i的值已经为10, 结果为十个10。

将var改为let,变量作用域不同,let作用在当前循环中,所以进入事件队列的定时器每次的i不同,最后打印结果会是 0 1 2...9。

宏任务 微任务

除了经常说的同步任务和异步任务之外,更可分为宏任务,微任务

主要宏任务:整http://www.cppcns.com段脚本scriptsetTimeoutsetTimeout...

主要微任务:promise.then...

执行流程:

1.整段脚本script作为宏任务开始执行

2.遇到微任务将其推入微任务队列,宏任务推入宏任务队列

3.宏任务执行完毕,检查有没有可执行的微任务

4.发现有可执行的微任务,将所有微任务执行完毕

5.开始新的宏任务,反复如此直到所有任务执行完毕

来一盘Promise

const p = new Promise(resolve => {
    console.log('a')
    resolve()
    console.log('b')
})

p.then(() => {
    console.log('c')
})

console.log('d')

结果:a b d c

1.整段script进入宏任务队列开始执行

2.promise创建立即执行,打印ab

3.遇到promise.then进入微任务队列

4.遇到console.log('d')打印d

5.整段代码作为宏任务执行完毕,有可执行的微任务,开始执行微任务,打印c。

setTimeout(function(){
    console.log('setTimeo编程客栈ut')
}, 0)

const p = new Promise(resolve => {
    console.log('a')
    resolve()
  http://www.cppcns.com  console.log('b')
})

p.then(() => {
    console.log('c')
})

console.log('d')

结果:a b d c setTimeout

1.setTimeout进入宏任务队列

2.promise创建立即执行,打印ab

3.遇到promise.then进入微任务队列

4.遇到console.log('d')打印d

5.有可执行的微任务,打印c

6.微任务执行完毕,开始执行新的宏任务,setTimeout开始执行,打印setTimeout

setTimeout(function(){
    console.log('setTimeout')
}, 0)

const p = new Promise(resolve => {
    console.log('a')
    resolve()
    console.log('b')
})

p.then(() => {
    console.log('c')
    setTimeout(function(){
        console.log('then中的setTimeout')
    }, 0)
})

console.log('d')

结果:a b d c setTimeout then中的setTimeout

1.同上

2.执行微任务打印c,遇到setTimeout将其推入宏任务队列中

3.定时器延迟时间相同,开始按照顺序执行宏任务,分别打印setTimeoutthen中的setTimeout

再加点定时器

console.log('a');

new Promise(resolve => {
    console.log('b')
    resolve()
}).then(() => {
    console.log('c')
    setTimeout(() => {
      console.log('d')
    }, 0)
})

setTimeout(() => {
    console.log('e')

用几道面试题来看JavaScript执行机制

扫一扫手机访问