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

熱AI工具

Undress AI Tool
免費(fèi)脫衣圖片

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

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

Clothoff.io
AI脫衣器

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

熱門文章

熱工具

記事本++7.3.1
好用且免費(fèi)的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強(qiáng)大的PHP整合開發(fā)環(huán)境

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

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

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

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

為了瀏覽器相容。 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」選項即可。

步驟: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中可以利用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ū)是語法錯誤,是指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中,可以利用array物件的length屬性來判斷數(shù)組裡總共有多少項,即取得數(shù)組中元素的個數(shù);該屬性可傳回數(shù)組中元素的數(shù)組,只需要使用「array.length」語句即可傳回表示數(shù)組物件的元素個數(shù)的數(shù)值,也就是長度值。

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