JavaScript中異步的示例分析-創(chuàng)新互聯(lián)

這篇文章將為大家詳細(xì)講解有關(guān)JavaScript中異步的示例分析,小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,希望大家閱讀完這篇文章后可以有所收獲。

主要從事網(wǎng)頁(yè)設(shè)計(jì)、PC網(wǎng)站建設(shè)(電腦版網(wǎng)站建設(shè))、wap網(wǎng)站建設(shè)(手機(jī)版網(wǎng)站建設(shè))、成都響應(yīng)式網(wǎng)站建設(shè)、程序開(kāi)發(fā)、微網(wǎng)站、小程序開(kāi)發(fā)等,憑借多年來(lái)在互聯(lián)網(wǎng)的打拼,我們?cè)诨ヂ?lián)網(wǎng)網(wǎng)站建設(shè)行業(yè)積累了豐富的成都做網(wǎng)站、網(wǎng)站制作、成都外貿(mào)網(wǎng)站建設(shè)、網(wǎng)絡(luò)營(yíng)銷經(jīng)驗(yàn),集策劃、開(kāi)發(fā)、設(shè)計(jì)、營(yíng)銷、管理等多方位專業(yè)化運(yùn)作于一體,具備承接不同規(guī)模與類型的建設(shè)項(xiàng)目的能力。

一、異步解決方案的進(jìn)化史
JavaScript的異步操作一直是個(gè)麻煩事,所以不斷有人提出它的各種解決方案??梢宰匪莸阶钤绲幕卣{(diào)函數(shù)(ajax老朋友),到Promise(不算新的朋友),再到ES6的Generator(強(qiáng)勁的朋友)。
幾年前我們可能用過(guò)一個(gè)比較著名的Async.js,但是它沒(méi)有擺脫回調(diào)函數(shù),并且錯(cuò)誤處理也是按照“回調(diào)函數(shù)的第一個(gè)參數(shù)用來(lái)傳遞錯(cuò)誤”這樣一個(gè)約定。而眾所周知的回調(diào)地獄仍然是一個(gè)比較突出的問(wèn)題,直到Generator改變了這種異步風(fēng)格。
但是ES7的async await的出現(xiàn)(碉堡的新朋友),我們可以輕松寫(xiě)出同步風(fēng)格的代碼同時(shí)又擁有異步機(jī)制,可以說(shuō)是目前最簡(jiǎn)單,最優(yōu)雅,最佳的解決方案了。

二、async await語(yǔ)法
async await語(yǔ)法比較簡(jiǎn)單,可以認(rèn)為是Generator的語(yǔ)法糖,比起星號(hào)和yield更具有語(yǔ)義化。下面一個(gè)簡(jiǎn)單的例子表示1秒之后輸出hello world:

function timeout(ms) {
 return new Promise((resolve) => {
  setTimeout(resolve, ms);
 });
}
async function asyncPrint(value, ms) {
 await timeout(ms);
 console.log(value)
}
asyncPrint('hello world', 1000);

await只能用在async函數(shù)中,如果用在普通函數(shù)就會(huì)報(bào)錯(cuò)

await后面跟的是一個(gè)Promise對(duì)象(當(dāng)然其它值也可以,但是會(huì)包裝成一個(gè)立即resolve的Promise,也就沒(méi)有意義了)

await會(huì)等待Promise的結(jié)果返回再繼續(xù)執(zhí)行

await等待的雖然是Promise對(duì)象,但是不必寫(xiě).then(),直接可以得到返回值,將上面的代碼微調(diào),發(fā)現(xiàn)返回值result也是可以輸出hello world:

function timeout(ms) {
 return new Promise((resolve) => {
  setTimeout(_ => {resolve('hello world')}, ms);
 });
}
async function asyncPrint(ms) {
 let result = await timeout(ms);
 console.log(result)
}
asyncPrint(1000);

三、async await錯(cuò)誤處理

前面說(shuō)了await等待的雖然是Promise對(duì)象,但是不必寫(xiě).then(),所以其實(shí)也不用寫(xiě).catch()了,直接用try catch就能捕捉錯(cuò)誤,這樣可以避免錯(cuò)誤處理代碼非常冗余和笨重,還是將上面的例子微調(diào):

function timeout(ms) {
 return new Promise((resolve, reject) => {
  setTimeout(_ => {reject('error')}, ms);//reject模擬出錯(cuò),返回error
 });
}
async function asyncPrint(ms) {
 try {
   console.log('start');
   await timeout(ms);//這里返回了錯(cuò)誤
   console.log('end');//所以這句代碼不會(huì)被執(zhí)行了
 } catch(err) {
   console.log(err); //這里捕捉到錯(cuò)誤error
 }
}
asyncPrint(1000);

如果有多個(gè)await,可以一起放在try catch中:

async function main() {
 try {
  const async1 = await firstAsync();
  const async2 = await secondAsync();
  const async3 = await thirdAsync();
 }
 catch (err) {
  console.error(err);
 }
}

四、async await注意點(diǎn)

1). 前面已經(jīng)說(shuō)過(guò),await命令后面的Promise對(duì)象,運(yùn)行結(jié)果很可能是reject或邏輯報(bào)錯(cuò),所以最好把a(bǔ)wait放在try catch代碼塊中。

2). 多個(gè)await命令的異步操作,如果不存在依賴關(guān)系,讓它們同時(shí)觸發(fā)。

const async1 = await firstAsync();
const async2 = await secondAsync();

上面代碼中,async1和async2如果是兩個(gè)獨(dú)立的異步操作,這樣寫(xiě)會(huì)比較耗時(shí),因?yàn)橹挥衒irstAsync完成以后,才會(huì)執(zhí)行secondAsync,完全可以用Promise.all優(yōu)雅地處理:

let [async1, async2] = await Promise.all([firstAsync(), secondAsync()]);

3). await只能用在async函數(shù)之中,如果用在普通函數(shù)就會(huì)報(bào)錯(cuò):

async function main() {
 let docs = [{}, {}, {}];
 //報(bào)錯(cuò) await is only valid in async function
 docs.forEach(function (doc) {
  await post(doc);
  console.log('main');
 });
}
function post(){
 return new Promise((resolve) => {
  setTimeout(resolve, 1000);
 });
}

在forEach內(nèi)部方法加上async就可以了:

async function main() {
 let docs = [{}, {}, {}];
 docs.forEach(async function (doc) {
  await post(doc);
  console.log('main');
 });
}
function post(){
 return new Promise((resolve) => {
  setTimeout(resolve, 1000);
 });
}

但是你會(huì)發(fā)現(xiàn)3個(gè)main是同時(shí)輸出的,這就說(shuō)明post是并發(fā)執(zhí)行的,而不是繼發(fā)執(zhí)行,改成for就可以解決問(wèn)題,3個(gè)main是分別相隔1秒輸出:

async function main() {
 let docs = [{}, {}, {}];
 for (let doc of docs) {
  await post(doc);
  console.log('main');
 }
}
function post(){
 return new Promise((resolve) => {
  setTimeout(resolve, 1000);
 });
}

關(guān)于“JavaScript中異步的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,使各位可以學(xué)到更多知識(shí),如果覺(jué)得文章不錯(cuò),請(qǐng)把它分享出去讓更多的人看到。

另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。

分享文章:JavaScript中異步的示例分析-創(chuàng)新互聯(lián)
當(dāng)前網(wǎng)址:http://bm7419.com/article20/dpocjo.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供面包屑導(dǎo)航、Google、企業(yè)建站、網(wǎng)站內(nèi)鏈手機(jī)網(wǎng)站建設(shè)、動(dòng)態(tài)網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

營(yíng)銷型網(wǎng)站建設(shè)