await操做符node
async function fun(){
var val = await 20;
console.log(val);
}
fun();
//20
async function bar(){
var val = await Promise.resolve(20);
console.log(val);
}
bar();
//20
複製代碼
async函數chrome
async function bar(name){
var subName = await Promise.resolve('fen');
return name + subName;
}
bar('li').then((res)=>{
console.log(res);
});
//lifen
複製代碼
new Promise((resolve) => {
resolve();
}).then(() => {
console.log('p1_1')
}).then(() => {
console.log('p1_2')
}).then(() => {
console.log('p1_3')
});
new Promise((resolve) => {
resolve();
}).then(() => {
console.log('p2_1')
}).then(() => {
console.log('p2_2')
}).then(() => {
console.log('p2_3')
});
複製代碼
結果以下:segmentfault
p1_1
p2_1
p1_2
p2_2
p1_3
p2_3
複製代碼
Promise是基於微任務的(微任務文章點擊連接),async/await可視爲 Promise 的語法糖,一樣基於微任務實現promise
EnquequeJob存放兩種類型的任務, 即PromiseResolveThenableJob和PromiseReactionJob, 而且都是屬於microtask類型的任務。瀏覽器
PromiseReactionJob: 能夠通俗的理解爲promise中的回調函數 PromiseResolveThenableJob(promiseToResolve, thenable, then): 建立和 promiseToResolve 關聯的 resolve function 和 reject function。以 then 爲調用函數,thenable 爲this,resolve function和reject function 爲參數調用返回。bash
Promise中resolve一個thenable對象,執行步驟如何? 在 Promise 中 resolve 一個 thenable 對象時,異步
這個過程須要做爲一個 job 加入微任務隊列,以保證對 then 方法的解析發生在其餘上下文代碼的解析以後async
let thenable = {
then(resolve, reject) {
console.log('in thenable');
resolve(100);
}
};
var promiseB = new Promise((resolve) => {
console.log('promiseB');
resolve(thenable);
});
//或var promiseB = Promise.resolve(thenable);
var promiseA = new Promise((resolve) => {
console.log('promiseA');
resolve();
});
promiseB.then((res) => {
console.log('out thenable ' + res)
}).then(() => {
console.log('then2')
}).then(() => {
console.log('then3')
});
promiseA.then(() => {
console.log(1);
}).then(() => {
console.log(2)
}).then(() => {
console.log(3)
});
複製代碼
結果以下:函數
promiseB
promiseA
in thenable
1
out thenable 100
2
then2
3
then3
複製代碼
in thenable 後於 promiseA 而先於 1 輸出,同時out thenable 100 在 1 後輸出。post
Promise.resolve(thenable)與new Promise(resolve=>(resolve(thenable)))的處理結果一致
正是因爲規範中對 thenable 的處理須要在一個微任務中完成,從而致使了第一個 Promise 的後續回調被延後了1個時序
var promiseA = Promise.resolve('promiseA');
var promiseB = new Promise((resolve) => {
resolve(promiseA);
});
//或var promiseB = Promise.resolve(promiseA);
promiseB.then(() => {
console.log('then1')
}).then(() => {
console.log('then2')
}).then(() => {
console.log('then3')
});
promiseA.then(() => {
console.log(1);
}).then(() => {
console.log(2);
}).then(() => {
console.log(3);
});
複製代碼
結果
//new Promise
1
2
then1
3
then2
then3
複製代碼
//採用或的結構Promise.resolve(promiseA)
then1
1
then2
2
then3
3
複製代碼
'then1'輸出晚了兩個時序。對於參數是一個promise ,resolve(promiseA)究竟如何工做的?
例子中,當promiseB被resolved的時候, 也就是將一個promise(代指A)當成另一個promise(代指B)的resolve參數,會向EnquequeJob插入一個PromiseResolveThenableJob任務。PromiseResolveThenableJob大概作了以下的事情:
() => {
promiseA.then(
resolvePromiseB,
rejectPromiseB
);
}
複製代碼
而且當resolvePromiseB執行後, promiseB的狀態才變成resolve,也就是B追隨A的狀態。
async function async1() {
console.log('async1')
await async2();
console.log('async1 end')
}
async function async2() {
console.log('async2');
}
async1();
new Promise(function (resolve) {
console.log('promise')
resolve();
}).then(function () {
console.log(1);
}).then(function () {
console.log(2);
}).then(function () {
console.log(3);
})
複製代碼
瀏覽器chrome76 環境中,結果以下:
//瀏覽器chrome76
async1
async2
promise
async1 end
1
2
3
複製代碼
node環境中,async1 end被推遲了2個時序
//node v10.6.0,
async1
async2
promise
1
2
async1 end
3
複製代碼
以瀏覽器運行結果來轉換
function async1(){
console.log('async1 start');
const p = async2();
return Promise.resolve(p).then(() => {
console.log('async1 end')
});
}
function async2(){
console.log('async2');
return Promise.resolve();
}
//其餘代碼同上
複製代碼
參考連接: 使人費解的 async/await 執行順序