亚洲国产日韩欧美一区二区三区,精品亚洲国产成人av在线,国产99视频精品免视看7,99国产精品久久久久久久成人热,欧美日韩亚洲国产综合乱

目錄
async/await是什麼
對比Promise,co,async/await
使用Promise鏈
co搭配生成器函數(shù)
async/await
深入async/await
async返回值
async函數(shù)的異常
并行
首頁 web前端 前端問答 async是es6還是es7的

async是es6還是es7的

Jan 29, 2023 pm 05:36 PM
es6 async await es7 async

async是es7的。 async和await是ES7中新增內(nèi)容,是對於非同步操作的解決方案;async/await可以說是co模組和生成器函數(shù)的語法糖,用更清晰的語意解決js非同步程式碼。 async顧名思義是「非同步」的意思,async用於聲明一個函數(shù)是異步的;async和await有一個嚴(yán)格規(guī)定,兩者都離不開對方,且await只能寫在async函數(shù)中。

async是es6還是es7的

本教學(xué)操作環(huán)境:windows7系統(tǒng)、ECMAScript 6版、Dell G3電腦。

ES7(ES2017)中提出的前端非同步特性:async、await。

async/await是什麼

async和await是ES7中新增內(nèi)容,對於非同步操作的解決方案,async/await可以說是co模組和生成器函數(shù)的語法糖。用更清晰的語意解決js異步程式碼。

async顧名思義是「非同步」的意思,async用來宣告一個函數(shù)是異步的。而await從字面意思上是「等待」的意思,就是用來等待非同步完成。

async和await它們兩有一個嚴(yán)格規(guī)定,兩者都離不開對方,但是,await只能寫在async函數(shù)中。

熟悉co模組的同學(xué)應(yīng)該都知道,co模組是TJ大神寫的一個使用生成器函數(shù)來解決非同步流程的模組,可以看做是生成器函數(shù)的執(zhí)行器。而async/await則是對co模組的升級,內(nèi)建生成器函數(shù)的執(zhí)行器,不再依賴co模組。同時,async返回的是Promise。

從上面來看,不管是co模組還是async/await,都是將Promise作為最基礎(chǔ)的單元,對Promise不很了解的同學(xué)可以先深入了解Promise。

對比Promise,co,async/await

#下面我們使用一個簡單的例子,來比較一下三種方式的異同,以及取捨。

我們採用mongodb的nodejs驅(qū)動,查詢mongodb資料庫作為例子,原因是mongodb的js驅(qū)動程式已經(jīng)預(yù)設(shè)實作了回傳Promise,而不用我們單獨(dú)去包裝Promise了。

使用Promise鏈

MongoClient.connect(url + db_name).then(db => {
    return db.collection('blogs');
}).then(coll => {
    return coll.find().toArray();
}).then(blogs => {
    console.log(blogs.length);
}).catch(err => {
    console.log(err);
})

Promise的then()方法可以回傳另一個Promise,也可以回傳一個同步的值,如果傳回的是一個同步值,將會被包裝成一個Promise。在上述的範(fàn)例中,db.collection()會傳回一個同步的值,即集合對象,但被包裝成Promise,將會透傳到下一個then()方法。上面一個例子,是使用的Promise鏈。先連接資料庫MongoClient.connect()回傳一個Promise,然後在then()方法裡取得資料庫物件db,然後再取得到coll物件再回傳。在下一個then()方法取得coll對象,然後進(jìn)行查詢,查詢結(jié)果返回,逐層呼叫then()方法,形成一個Promise鏈。在這個Promise鏈上,如果任何一個環(huán)節(jié)出現(xiàn)異常,就會被最後的catch()捕捉到??梢哉f,這個使用Promise鏈寫的程式碼,比層層呼叫回呼函數(shù)更優(yōu)雅,流程也更明確。先取得資料庫對象,再取得集合對象,最後查詢資料。但這裡有個不太「優(yōu)雅」的問題,在於,每一個then()方法取得的對象,都是上一個then()方法返回的資料。而不能跨層訪問。什麼意思,就是說在第三個then(blogs => {})中我們只能取得到查詢的結(jié)果blogs,而不能使用上面的db物件和coll物件。這時候,如果要列印blogs清單後,要關(guān)閉資料庫db.close()怎麼辦?這個時候,可以兩種解決方法:

第一種是,使用then()巢狀。我們將Promise鏈打斷,使之嵌套,猶如使用回調(diào)函數(shù)的嵌套一般:

MongoClient.connect(url + db_name).then(db => {
    let coll = db.collection('blogs');
    coll.find().toArray().then(blogs => {
        console.log(blogs.length);
        db.close();
    }).catch(err => {
        console.log(err);
    });
}).catch(err => {
    console.log(err);
})

這裡我們將兩個Promise嵌套,這樣在最後一個查詢操作裡面,就可以調(diào)用外面的db對象了。但這中方式,並不推薦。原因很簡單,我們從一種回呼函數(shù)地獄走向了另一種Promise回調(diào)地獄。
而且,我們要對每個Promise的異常進(jìn)行捕捉,因為Promise沒有形成鏈。

還有一種方式, 是在每個then()方法裡都會將db傳過來:

MongoClient.connect(url + db_name).then(db => {
    return {db:db,coll:db.collection('blogs')};
}).then(result => {
    return {db:result.db,blogs:result.coll.find().toArray()};
}).then(result => {
    return result.blogs.then(blogs => {   //注意這里,result.coll.find().toArray()返回的是一個Promise,因此這里需要再解析一層
        return {db:result.db,blogs:blogs}
    })
}).then(result => {
    console.log(result.blogs.length);
    result.db.close();
}).catch(err => {
    console.log(err);
});

我們在每個then()方法的回傳中,都會將db及其每次的其他結(jié)果組成一個物件回傳。請注意,如果每次的結(jié)果都是一個同步的值還好說,但是如果是一個Promise值,每一個Promise都需要多做一層解析。
例如上面的一個例子,第二個then()方法傳回的{db:result.db,blogs:result.coll.find().toArray()}物件中,blogs是一個Promise,在下一個then()方法中,我們無法直接引用部落格清單陣列值,因此需要先呼叫then()方法解析一層,然後將兩個同步值db和blogs傳回。請注意,這裡涉及了Promise的嵌套,不過一個Promise只嵌套一層then()。

這種方式,也是很蛋疼的一個方式,因為如果遇到then()方法中返回的不是同步的值,而是Promise的話,我們需要多做很多工作。而且,每次都透傳一個“多余”的db對象,在邏輯上也有點(diǎn)冗余。

但除此之外,對于Promise鏈的使用,如果遇到上面的問題,好像也沒其他更好的方法解決了。我們只能根據(jù)場景去選擇一種“最優(yōu)”的方案,如果要使用Promise鏈的話。

鑒于Promise上面蛋疼的問題,TJ大神將ES6中的生成器函數(shù),用co模塊包裝了一下,以更優(yōu)雅的方式來解決上面的問題。

co搭配生成器函數(shù)

如果使用co模塊搭配生成器函數(shù),那么上面的例子可以改寫如下:

const co = require('co');
co(function* (){
    let db = yield MongoClient.connect(url + db_name);
    let coll = db.collection('blogs');
    let blogs = yield coll.find().toArray();
    console.log(blogs.length);
    db.close();
}).catch(err => {
    console.log(err);
});

co是一個函數(shù),將接受一個生成器函數(shù)作為參數(shù),去執(zhí)行這個生成器函數(shù)。生成器函數(shù)中使用yield關(guān)鍵字來“同步”獲取每個異步操作的值。
上面代碼在代碼形式上,比上面使用Promise鏈要優(yōu)雅,我們消滅了回調(diào)函數(shù),代碼看起來都是同步的。除了使用co和yield有點(diǎn)怪之外。

使用co模塊,我們要將所有的操作包裝成一個生成器函數(shù),然后使用co()去調(diào)用這個生成器函數(shù)??瓷先ヒ策€可以接受,但是ES的進(jìn)化是不滿足于此的,于是async/await被提到了ES7的提案。

async/await

我們先看一下使用async/await改寫上面的代碼:

(async function(){
    let db = await MongoClient.connect(url + db_name);
    let coll = db.collection('blogs');
    let blogs = await coll.find().toArray();
    console.log(blogs.length);
    db.close();
})().catch(err => {
    console.log(err);
});

我們對比代碼可以看出,async/await和co兩種方式代碼極為相似。co換成了async,yield換成了await。同時生成器函數(shù)變成了普通函數(shù)。這種方式在語義上更加清晰明了,async表明這個函數(shù)是異步的,同時await表示要“等待”異步操作返回值。

async函數(shù)返回一個Promise,上面的代碼其實是這樣:

let getBlogs = async function(){
    let db = await MongoClient.connect(url + db_name);
    let coll = db.collection('blogs');
    let blogs = await coll.find().toArray();
    db.close();
    return blogs;
};
getBlogs().then(result => {
    console.log(result.length);
}).catch(err => {
    console.log(err);
})

我們定義getBlogs為一個async函數(shù),最后返回得到的博客列表最終會被包裝成一個Promise返回,如上,我們直接調(diào)用getBlogs().then()方法可獲取async函數(shù)返回值。

好了,上面我們簡單對比了一下三種解決異步方案,下面我們來深入了解一下async/await。

深入async/await

async返回值

async用于定義一個異步函數(shù),該函數(shù)返回一個Promise。
如果async函數(shù)返回的是一個同步的值,這個值將被包裝成一個理解resolve的Promise,等同于return Promise.resolve(value)。
await用于一個異步操作之前,表示要“等待”這個異步操作的返回值。await也可以用于一個同步的值。

//返回一個Promise
let timer = async function timer(){
    return new Promise((resolve,reject) => {
        setTimeout(() => {
            resolve('500');
        },500);
    });
}
timer().then(result => {
  console.log(result);  //500
}).catch(err => {
    console.log(err.message);
});
//返回一個同步的值
let sayHi = async function sayHi(){
  let hi = await 'hello world';   
  return hi;  //等同于return Promise.resolve(hi);
}
sayHi().then(result => {
  console.log(result);
});

上面這個例子返回是一個同步的值,字符串’hello world’,sayHi()是一個async函數(shù),返回值被包裝成一個Promise,可以調(diào)用then()方法獲取返回值。對于一個同步的值,可以使用await,也可以不使用await。效果效果是一樣的。具體用不用,看情況。

比如上面使用mongodb查詢博客那個例子,let coll = db.collection('blogs');,這里我們就沒有用await,因為這是一個同步的值。當(dāng)然,也可以使用await,這樣會顯得代碼統(tǒng)一。雖然效果是一樣的。

async函數(shù)的異常

let sayHi = async function sayHi(){
    throw new Error('出錯了');
}
sayHi().then(result => {
  console.log(result);
}).catch(err => {
    console.log(err.message);   //出錯了
});

我們直接在async函數(shù)中拋出一個異常,由于返回的是一個Promise,因此,這個異常可以調(diào)用返回Promise的catch()方法捕捉到。

和Promise鏈的對比:
我們的async函數(shù)中可以包含多個異步操作,其異常和Promise鏈有相同之處,如果有一個Promise被reject()那么后面的將不會再進(jìn)行。

let count = ()=>{
    return new Promise((resolve,reject) => {
        setTimeout(()=>{
            reject('故意拋出錯誤');
        },500);
    });
}
let list = ()=>{
    return new Promise((resolve,reject)=>{
        setTimeout(()=>{
            resolve([1,2,3]);
        },500);
    });
}
let getList = async ()=>{
    let c = await count();
    let l = await list();
    return {count:c,list:l};
}
console.time('begin');
getList().then(result => {
    console.log(result);
}).catch(err => {
    console.timeEnd('begin');
    console.log(err);
});
//begin: 507.490ms
//故意拋出錯誤

如上面的代碼,定義兩個異步操作,count和list,使用setTimeout延時500毫秒,count故意直接拋出異常,從輸出結(jié)果來看,count()拋出異常后,直接由catch()捕捉到了,list()并沒有繼續(xù)執(zhí)行。

并行

使用async后,我們上面的例子都是串行的。比如上個list()和count()的例子,我們可以將這個例子用作分頁查詢數(shù)據(jù)的場景。先查詢出數(shù)據(jù)庫中總共有多少條記錄,然后再根據(jù)分頁條件查詢分頁數(shù)據(jù),最后返回分頁數(shù)據(jù)以及分頁信息。

我們上面的例子count()和list()有個“先后順序”,即我們先查的總數(shù),然后又查的列表。其實,這兩個操作并無先后關(guān)聯(lián)性,我們可以異步的同時進(jìn)行查詢,然后等到所有結(jié)果都返回時再拼裝數(shù)據(jù)即可。

let count = ()=>{
    return new Promise((resolve,reject) => {
        setTimeout(()=>{
            resolve(100);
        },500);
    });
}
let list = ()=>{
    return new Promise((resolve,reject)=>{
        setTimeout(()=>{
            resolve([1,2,3]);
        },500);
    });
}
let getList = async ()=>{
    let result = await Promise.all([count(),list()]);
    return result;
}
console.time('begin');
getList().then(result => {
    console.timeEnd('begin');  //begin: 505.557ms
    console.log(result);       //[ 100, [ 1, 2, 3 ] ]
}).catch(err => {
    console.timeEnd('begin');
    console.log(err);
});

我們將count()和list()使用Promise.all()“同時”執(zhí)行,這里count()和list()可以看作是“并行”執(zhí)行的,所耗時間將是兩個異步操作中耗時最長的耗時。

最後得到的結(jié)果是兩個運(yùn)算的結(jié)果所組成的陣列。我們只需要按照順序取出數(shù)組中的值。

【推薦學(xué)習(xí):javascript影片教學(xué)

#

以上是async是es6還是es7的的詳細(xì)內(nèi)容。更多資訊請關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

本網(wǎng)站聲明
本文內(nèi)容由網(wǎng)友自願投稿,版權(quán)歸原作者所有。本站不承擔(dān)相應(yīng)的法律責(zé)任。如發(fā)現(xiàn)涉嫌抄襲或侵權(quán)的內(nèi)容,請聯(lián)絡(luò)admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費(fèi)脫衣圖片

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅(qū)動的應(yīng)用程序,用於創(chuàng)建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費(fèi)的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費(fèi)的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強(qiáng)大的PHP整合開發(fā)環(huán)境

Dreamweaver CS6

Dreamweaver CS6

視覺化網(wǎng)頁開發(fā)工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

ES6怎麼求數(shù)組反轉(zhuǎn) ES6怎麼求數(shù)組反轉(zhuǎn) Oct 26, 2022 pm 06:19 PM

在ES6中,可以利用數(shù)組物件的reverse()方法來實現(xiàn)數(shù)組反轉(zhuǎn),該方法用於顛倒數(shù)組中元素的順序,將最後一個元素放在第一位,而第一個元素放在最後,語法「array.reverse()」。 reverse()方法會修改原始數(shù)組,如果不想修改需要配合擴(kuò)充運(yùn)算子「...」使用,語法「[...array].reverse()」。

async是es6還是es7的 async是es6還是es7的 Jan 29, 2023 pm 05:36 PM

async是es7的。 async和await是ES7中新增內(nèi)容,是對於非同步操作的解決方案;async/await可以說是co模組和生成器函數(shù)的語法糖,用更清晰的語意解決js非同步程式碼。 async顧名思義是「非同步」的意思,async用於聲明一個函數(shù)是異步的;async和await有一個嚴(yán)格規(guī)定,兩者都離不開對方,且await只能寫在async函數(shù)中。

小程式為什麼要將es6轉(zhuǎn)es5 小程式為什麼要將es6轉(zhuǎn)es5 Nov 21, 2022 pm 06:15 PM

為了瀏覽器相容。 ES6作為JS的新規(guī)範(fàn),加入了許多新的語法和API,但現(xiàn)代瀏覽器對ES6新特性支援不高,所以需將ES6程式碼轉(zhuǎn)換為ES5程式碼。在微信web開發(fā)者工具中,會預(yù)設(shè)使用babel將開發(fā)者ES6語法程式碼轉(zhuǎn)換為三端都能很好支援的ES5的程式碼,幫助開發(fā)者解決環(huán)境不同所帶來的開發(fā)問題;只需要在專案中配置勾選好「ES6轉(zhuǎn)ES5」選項即可。

es6怎麼找出2個數(shù)組中不同項 es6怎麼找出2個數(shù)組中不同項 Nov 01, 2022 pm 06:07 PM

步驟:1、將兩個陣列分別轉(zhuǎn)換為set類型,語法「newA=new Set(a);newB=new Set(b);」;2、利用has()和filter()求差集,語法“ new Set([...newA].filter(x =>!newB.has(x)))”,差集元素會被包含在一個set集合中返回;3、利用Array.from將集合轉(zhuǎn)為數(shù)組類型,語法“Array.from(集合)”。

es5和es6怎麼實現(xiàn)陣列去重 es5和es6怎麼實現(xiàn)陣列去重 Jan 16, 2023 pm 05:09 PM

es5中可以利用for語句和indexOf()函數(shù)來實現(xiàn)數(shù)組去重,語法“for(i=0;i<數(shù)組長度;i++){a=newArr.indexOf(arr[i]);if(a== -1){...}}」。在es6中可以利用擴(kuò)充運(yùn)算子、Array.from()和Set來去重;需要先將陣列轉(zhuǎn)為Set物件來去重,然後利用擴(kuò)充運(yùn)算子或Array.from()函數(shù)來將Set物件轉(zhuǎn)回數(shù)組即可。

es6暫時性死區(qū)是什麼意思 es6暫時性死區(qū)是什麼意思 Jan 03, 2023 pm 03:56 PM

在es6中,暫時性死區(qū)是語法錯誤,是指let和const命令使區(qū)塊形成封閉的作用域。在程式碼區(qū)塊內(nèi),使用let/const指令宣告變數(shù)之前,變數(shù)都是不可用的,在變數(shù)宣告之前屬於該變數(shù)的「死區(qū)」;這在語法上,稱為「暫時性死區(qū)」。 ES6規(guī)定暫時性死區(qū)和let、const語句不出現(xiàn)變量提升,主要是為了減少運(yùn)行時錯誤,防止在變量聲明前就使用這個變量,從而導(dǎo)致意料之外的行為。

es6怎麼判斷陣列裡總共有多少項 es6怎麼判斷陣列裡總共有多少項 Jan 18, 2023 pm 07:22 PM

在es6中,可以利用array物件的length屬性來判斷數(shù)組裡總共有多少項,即取得數(shù)組中元素的個數(shù);該屬性可傳回數(shù)組中元素的數(shù)組,只需要使用「array.length」語句即可傳回表示數(shù)組物件的元素個數(shù)的數(shù)值,也就是長度值。

es6 import會變數(shù)提升嗎 es6 import會變數(shù)提升嗎 Jan 18, 2023 pm 07:44 PM

ES6 import會產(chǎn)生變數(shù)提升的現(xiàn)象。變數(shù)提升是將變數(shù)宣告提升到它所在作用域的最開始的部分。 js要經(jīng)歷編譯跟執(zhí)行階段,在編譯階段的時候,會蒐集所有的變量聲明並且提前聲明變量,而其他的語句都不會改變他們的順序,因此,在編譯階段的時候,第一步就已經(jīng)執(zhí)行了,而第二部則是在執(zhí)行階段執(zhí)行到該語句的時候才執(zhí)行。

See all articles