Promise 学习2
这是第二段代码
new Promise((resolve, reject) => {
console.log("外部promise");
resolve();
})
.then(() => {
console.log("外部第一个then");
new Promise((resolve, reject) => {
console.log("内部promise");
resolve();
})
.then(() => {
console.log("内部第一个then");
})
.then(() => {
console.log("内部第二个then");
});
})
.then(() => {
console.log("外部第二个then");
});
复制代码
这段代码和第一段代码就相差一个 return ,然而结果确是不一样的。
那这个怎么理解呢?
我们核心要看 then 的回调函数是啥时候注册的,我们知道,事件机制是 “先注册先执行”,即数据结构中的 “队列” 的模式,first in first out。那么重点我们来看下他们谁先注册的。
外部的第二个 then 的注册,需要等待 外部的第一个 then 的同步代码执行完成。 当执行内部的 new Promise 的时候,然后碰到 resolve,resolve 执行完成,代表此时的该 Promise 状态已经扭转,之后开始内部的第一个 .then 的微任务的注册,此时同步执行完成。我们知道需要执行的动作是一个微任务,那么自然要先执行完同步任务,比如如下:
new Promise((resolve, reject) => {
resolve();
console.log(111);
})
.then(() => {
consle.log(222);
})
复制代码
这个代码显然优先输出执行 1111,再执行 222。 因为 222 的输出是微任务的执行,111 是同步执行。
同理回到上面的代码,内部的 resolve 之后,当然是先执行内部的 new Promise 的第一个 then 的注册,这个 new Promise 执行完成,立即同步执行了后面的 .then 的注册。
然而这个内部的第二个 then 是需要第一个 then 的的执行完成来决定的,而第一个 then 的回调是没有执行,仅仅只是执行了同步的 .then 方法的注册,所以会进入等待状态。
这个时候,外部的第一个 then 的同步操作已经完成了,然后开始注册外部的第二个 then,此时外部的同步任务也都完成了。同步操作完成之后,那么开始执行微任务,我们发现 内部的第一个 then 是优先于外部的第二个 then 的注册,所以会执行完内部的第一个 then 之后,然后注册内部的第二个 then ,然后执行外部的第二个 then ,然后再执行内部的第二个 then。
output:
外部promise
外部第一个then
内部promise
内部第一个then
外部第二个then
内部第二个then
我们发现,这里显然是执行完一个 then ,接着会注册该 then 之后的下一个 then,按照任务队列的原理,我们可以发现,内外 then 是交替执行,然后交替注册的。所以才会出现输出内外交替内容。
另外,我这里所说的 then 的注册,是指微任务队列的注册,并不是 .then 的方法的执行,实际上 .then 方法的执行,我们可以理解为仅仅只是初始化而已。如果看过源码的会知道,.then 的执行确实是同步的,内部是再开启一个 new Promise ,但是由于上一个状态未流转,该 then 并不会此时注册到微任务队列中,而是会等待上一个的执行完成,所以我们把 .then 没注册微任务就理解成尚没执行是没有问题的。