相關推薦:《javascript影片教學》
#async / await是ES7的重要特性之一,也是目前社群裡公認的優秀異步解決方案。目前,async / await這個特性已經是stage 3的建議,可以看看TC39的進度,本篇文章將分享async / await是如何運作的,閱讀本文前,希望你具備Promise、generator、yield等ES6的相關知識。
在詳細介紹async / await之前,先回顧下目前在ES6中比較好的非同步處理辦法。下面的範例中資料請求以Node.js中的request模組,資料介面採用Github v3 api文件提供的repo程式碼倉庫詳情API作為範例示範。
Promise對非同步的處理
雖然Node.js的非同步IO帶來了對高並發的良好支持,同時也讓「回調」成為災難,很容易造成回調地獄。傳統的方式例如使用具名函數,雖然可以減少巢狀的層數,讓程式碼看起來比較清晰。但是會造成比較差的編碼和調試體驗,你需要經常使用用ctrl f去尋找某個具名函數的定義,這使得IDE視窗經常上下來回跳動。使用Promise之後,可以很好的減少嵌套的層數。另外Promise的實作採用了狀態機,在函數裡面可以很好的透過resolve和reject進行流程控制,你可以依照順序鍊式的去執行一系列程式碼邏輯了。以下是使用Promise的範例:
const request = require('request'); // 请求的url和header const options = { url: 'https://api.github.com/repos/cpselvis/zhihu-crawler', headers: { 'User-Agent': 'request' } }; // 获取仓库信息 const getRepoData = () => { return new Promise((resolve, reject) => { request(options, (err, res, body) => { if (err) { reject(err); } resolve(body); }); }); }; getRepoData() .then((result) => console.log(result);) .catch((reason) => console.error(reason);); // 此处如果是多个Promise顺序执行的话,如下: // 每个then里面去执行下一个promise // getRepoData() // .then((value2) => {return promise2}) // .then((value3) => {return promise3}) // .then((x) => console.log(x))
不過Promise仍然有缺陷,它只是減少了嵌套,並不能完全消除嵌套。舉個例子,對於多個promise串列執行的情況,第一個promise的邏輯執行完之後,我們需要在它的then函數裡面去執行第二個promise,這個時候會產生一層巢狀。另外,採用Promise的程式碼看起來依然是異步的,如果寫的程式碼如果能夠變成同步該有多好!
Generator對非同步的處理
談到generator,你應該不會對它感到陌生。在Node.js中對於回呼的處理,我們常用的TJ / Co就是使用generator結合promise來實現的,co是coroutine的簡稱,藉鑑於python、lua等語言中的協程。它可以將非同步的程式碼邏輯寫成同步的方式,這使得程式碼的閱讀和組織變得更加清晰,也便於偵錯。
const co = require('co'); const request = require('request'); const options = { url: 'https://api.github.com/repos/cpselvis/zhihu-crawler', headers: { 'User-Agent': 'request' } }; // yield后面是一个生成器 generator const getRepoData = function* () { return new Promise((resolve, reject) => { request(options, (err, res, body) => { if (err) { reject(err); } resolve(body); }); }); }; co(function* () { const result = yield getRepoData; // ... 如果有多个异步流程,可以放在这里,比如 // const r1 = yield getR1; // const r2 = yield getR2; // const r3 = yield getR3; // 每个yield相当于暂停,执行yield之后会等待它后面的generator返回值之后再执行后面其它的yield逻辑。 return result; }).then(function (value) { console.log(value); }, function (err) { console.error(err.stack); });
async / await對非同步的處理
雖然co是社區裡面的優秀非同步解決方案,但並不是語言標準,只是過渡方案。 ES7語言層面提供async / await去解決語言層面的難題。目前async / await 在 IE edge已經可以直接使用了,但chrome和Node.js還沒有支援。幸運的是,babel已經支援async的transform了,所以我們使用的時候引入babel就行。在開始之前我們需要介紹以下的package,preset-stage-3裡就有我們需要的async/await的編譯檔。
無論是在Browser或Node.js端都需要安裝下面的套件。
$ npm install babel-core --save $ npm install babel-preset-es2015 --save $ npm install babel-preset-stage-3 --save
這裡推薦使用babel官方提供的require hook方法。就是透過require進來後,接下來的檔案進行require的時候都會經過Babel的處理。因為我們知道CommonJs是同步的模組依賴,所以也是可行的方法。這個時候,需要寫兩個文件,一個是啟動的js文件,另一個是真正執行程式的js文件。
啟動檔index.js
require('babel-core/register'); require('./async.js');
真正執行程式的async.js
const request = require('request'); const options = { url: 'https://api.github.com/repos/cpselvis/zhihu-crawler', headers: { 'User-Agent': 'request' } }; const getRepoData = () => { return new Promise((resolve, reject) => { request(options, (err, res, body) => { if (err) { reject(err); } resolve(body); }); }); }; async function asyncFun() { try { const value = await getRepoData(); // ... 和上面的yield类似,如果有多个异步流程,可以放在这里,比如 // const r1 = await getR1(); // const r2 = await getR2(); // const r3 = await getR3(); // 每个await相当于暂停,执行await之后会等待它后面的 //函数(不是generator)返回值之后再执行后面其它的await逻辑。 return value; } catch (err) { console.log(err); } } asyncFun().then(x => console.log(`x: ${x}`)).catch(err => console.error(err));
注意點:
- async用來申明裡麵包裹的內容可以進行同步的方式執行,await則是執行順序控制,每次執行一個await,程式都會暫停等待await回傳值,然後再執行之後的await。
- await後面呼叫的函數需要回傳一個promise,另外這個函數是一個普通的函數即可,而不是generator。
- await只能用在async函數之中,用在普通函數中會報錯。
- await指令後面的 Promise 對象,執行結果可能是 rejected,所以最好把 await 指令放在 try...catch 程式碼區塊中。
其實,async / await的用法和co差不多,await和yield都是表示暫停,外麵包裹一層async 或者 co來表示裡面的程式碼可以採用同步的方式進行處理。不過async / await裡面的await後面跟著的函數不需要額外處理,co是需要將它寫成一個generator的。
更多程式相關知識,請造訪:程式設計學習! !
以上是詳解Javascript中async/await是如何運作的的詳細內容。更多資訊請關注PHP中文網其他相關文章!

不同JavaScript引擎在解析和執行JavaScript代碼時,效果會有所不同,因為每個引擎的實現原理和優化策略各有差異。 1.詞法分析:將源碼轉換為詞法單元。 2.語法分析:生成抽象語法樹。 3.優化和編譯:通過JIT編譯器生成機器碼。 4.執行:運行機器碼。 V8引擎通過即時編譯和隱藏類優化,SpiderMonkey使用類型推斷系統,導致在相同代碼上的性能表現不同。

JavaScript在現實世界中的應用包括服務器端編程、移動應用開發和物聯網控制:1.通過Node.js實現服務器端編程,適用於高並發請求處理。 2.通過ReactNative進行移動應用開發,支持跨平台部署。 3.通過Johnny-Five庫用於物聯網設備控制,適用於硬件交互。

我使用您的日常技術工具構建了功能性的多租戶SaaS應用程序(一個Edtech應用程序),您可以做同樣的事情。 首先,什麼是多租戶SaaS應用程序? 多租戶SaaS應用程序可讓您從唱歌中為多個客戶提供服務

本文展示了與許可證確保的後端的前端集成,並使用Next.js構建功能性Edtech SaaS應用程序。 前端獲取用戶權限以控制UI的可見性並確保API要求遵守角色庫

JavaScript是現代Web開發的核心語言,因其多樣性和靈活性而廣泛應用。 1)前端開發:通過DOM操作和現代框架(如React、Vue.js、Angular)構建動態網頁和單頁面應用。 2)服務器端開發:Node.js利用非阻塞I/O模型處理高並發和實時應用。 3)移動和桌面應用開發:通過ReactNative和Electron實現跨平台開發,提高開發效率。

JavaScript的最新趨勢包括TypeScript的崛起、現代框架和庫的流行以及WebAssembly的應用。未來前景涵蓋更強大的類型系統、服務器端JavaScript的發展、人工智能和機器學習的擴展以及物聯網和邊緣計算的潛力。

JavaScript是現代Web開發的基石,它的主要功能包括事件驅動編程、動態內容生成和異步編程。 1)事件驅動編程允許網頁根據用戶操作動態變化。 2)動態內容生成使得頁面內容可以根據條件調整。 3)異步編程確保用戶界面不被阻塞。 JavaScript廣泛應用於網頁交互、單頁面應用和服務器端開發,極大地提升了用戶體驗和跨平台開發的靈活性。

Python更适合数据科学和机器学习,JavaScript更适合前端和全栈开发。1.Python以简洁语法和丰富库生态著称,适用于数据分析和Web开发。2.JavaScript是前端开发核心,Node.js支持服务器端编程,适用于全栈开发。


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

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

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

MantisBT
Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

MinGW - Minimalist GNU for Windows
這個專案正在遷移到osdn.net/projects/mingw的過程中,你可以繼續在那裡關注我們。 MinGW:GNU編譯器集合(GCC)的本機Windows移植版本,可自由分發的導入函式庫和用於建置本機Windows應用程式的頭檔;包括對MSVC執行時間的擴展,以支援C99功能。 MinGW的所有軟體都可以在64位元Windows平台上運作。

ZendStudio 13.5.1 Mac
強大的PHP整合開發環境

EditPlus 中文破解版
體積小,語法高亮,不支援程式碼提示功能

禪工作室 13.0.1
強大的PHP整合開發環境