国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

把 Node.js 中的回調(diào)轉(zhuǎn)換為 Promise

開發(fā) 前端
在本文中,我們將討論如何將 JavaScript 回調(diào)轉(zhuǎn)換為 Promise。ES6 的知識(shí)將會(huì)派上用場(chǎng),因?yàn)槲覀儗?huì)使用 展開操作符之類的功能來簡(jiǎn)化要做的事情。

介紹

在幾年前,回調(diào)是 JavaScript 中實(shí)現(xiàn)執(zhí)行異步代碼的唯一方法?;卣{(diào)本身幾乎沒有什么問題,最值得注意的是“回調(diào)地獄”。

在 ES6 中引入了 Promise 作為這些問題的解決方案。最后通過引入 async/await 關(guān)鍵字來提供更好的體驗(yàn)并提高了可讀性。

即使有了新的方法,但是仍然有許多使用回調(diào)的原生模塊和庫(kù)。在本文中,我們將討論如何將 JavaScript 回調(diào)轉(zhuǎn)換為 Promise。ES6 的知識(shí)將會(huì)派上用場(chǎng),因?yàn)槲覀儗?huì)使用 展開操作符之類的功能來簡(jiǎn)化要做的事情。

什么是回調(diào)

回調(diào)是一個(gè)函數(shù)參數(shù),恰好是一個(gè)函數(shù)本身。雖然我們可以創(chuàng)建任何函數(shù)來接受另一個(gè)函數(shù),但回調(diào)主要用于異步操作。

JavaScript 是一種解釋性語言,一次只能處理一行代碼。有些任務(wù)可能需要很長(zhǎng)時(shí)間才能完成,例如下載或讀取大文件等。JavaScript 將這些運(yùn)行時(shí)間很長(zhǎng)的任務(wù)轉(zhuǎn)移到瀏覽器或 Node.js 環(huán)境中的其他進(jìn)程中。這樣它就不會(huì)阻止其他代碼的執(zhí)行。

通常異步函數(shù)會(huì)接受回調(diào)函數(shù),所以完成之后可以處理其數(shù)據(jù)。

舉個(gè)例子,我們將編寫一個(gè)回調(diào)函數(shù),這個(gè)函數(shù)會(huì)在程序成功從硬盤讀取文件之后執(zhí)行。

所以需要準(zhǔn)備一個(gè)名為 sample.txt 的文本文件,其中包含以下內(nèi)容:

  1. Hello world from sample.txt 

然后寫一個(gè)簡(jiǎn)單的 Node.js 腳本來讀取文件:

  1. const fs = require('fs'); 
  2.  
  3. fs.readFile('./sample.txt', 'utf-8', (err, data) => { 
  4.     if (err) { 
  5.         // 處理錯(cuò)誤 
  6.         console.error(err); 
  7.           return; 
  8.     } 
  9.     console.log(data); 
  10. }); 
  11.  
  12. for (let i = 0; i < 10; i++) { 
  13.     console.log(i); 

運(yùn)行代碼后將會(huì)輸出:

  1. ... 
  2. Hello world from sample.txt 

如果這段代碼,應(yīng)該在執(zhí)行回調(diào)之前看到 0..9 被輸出到控制臺(tái)。這是因?yàn)?JavaScript 的異步管理機(jī)制。在讀取文件完畢之后,輸出文件內(nèi)容的回調(diào)才被調(diào)用。

順便說明一下,回調(diào)也可以在同步方法中使用。例如 Array.sort() 會(huì)接受一個(gè)回調(diào)函數(shù),這個(gè)函數(shù)允許你自定義元素的排序方式。

接受回調(diào)的函數(shù)被稱為“高階函數(shù)”。

現(xiàn)在我們有了一個(gè)更好的回調(diào)方法。那么們繼續(xù)看看什么是 Promise。

什么是 Promise

在 ECMAScript 2015(ES6)中引入了 Promise,用來改善在異步編程方面的體驗(yàn)。顧名思義,JavaScript 對(duì)象最終將返回的“值”或“錯(cuò)誤”應(yīng)該是一個(gè) Promise。

一個(gè) Promise 有 3 個(gè)狀態(tài):

  • Pending(待處理):用來指示異步操作尚未完成的初始狀態(tài)。
  • Fulfilled(已完成):表示異步操作已成功完成。
  • Rejected(拒絕):表示異步操作失敗。

大多數(shù) Promise 最終看起來像這樣:

  1. someAsynchronousFunction() 
  2.     .then(data => { 
  3.         // promise 被完成 
  4.         console.log(data); 
  5.     }) 
  6.     .catch(err => { 
  7.         // promise 被拒絕 
  8.         console.error(err); 
  9.     }); 

Promise 在現(xiàn)代 JavaScript 中非常重要,因?yàn)樗鼈兣c ECMAScript 2016 中引入的 async/await 關(guān)鍵字一起使用。使用 async / await 就不需要再用回調(diào)或 then() 和 catch() 來編寫異步代碼。

如果要改寫前面的例子,應(yīng)該是這樣:

  1. try { 
  2.     const data = await someAsynchronousFunction(); 
  3. } catch(err) { 
  4.     // promise 被拒絕 
  5.     console.error(err); 

這看起來很像“一般的”同步 JavaScript。大多數(shù)流行的JavaScript庫(kù)和新項(xiàng)目都把 Promises 與 async/await 關(guān)鍵字放在一起用。

但是,如果你要更新現(xiàn)有的庫(kù)或遇到舊的代碼,則可能會(huì)對(duì)將基于回調(diào)的 API 遷移到基于 Promise 的 API 感興趣,這樣可以改善你的開發(fā)體驗(yàn)。

來看一下將回調(diào)轉(zhuǎn)換為 Promise 的幾種方法。

將回調(diào)轉(zhuǎn)換為 Promise

Node.js Promise

大多數(shù)在 Node.js 中接受回調(diào)的異步函數(shù)(例如 fs 模塊)有標(biāo)準(zhǔn)的實(shí)現(xiàn)方式:把回調(diào)作為最后一個(gè)參數(shù)傳遞。

例如這是在不指定文本編碼的情況下用 fs.readFile() 讀取文件的方法:

  1. fs.readFile('./sample.txt', (err, data) => { 
  2.     if (err) { 
  3.         console.error(err); 
  4.           return; 
  5.     } 
  6.     console.log(data); 
  7. }); 

注意:如果你指定 utf-8 作為編碼,那么得到的輸出是一個(gè)字符串。如果不指定得到的輸出是 Buffer。

另外傳給這個(gè)函數(shù)的回調(diào)應(yīng)接受 Error,因?yàn)樗堑谝粋€(gè)參數(shù)。之后可以有任意數(shù)量的輸出。

如果你需要轉(zhuǎn)換為 Promise 的函數(shù)遵循這些規(guī)則,那么可以用 util.promisify ,這是一個(gè)原生 Node.js 模塊,其中包含對(duì) Promise 的回調(diào)。

首先導(dǎo)入ʻutil`模塊:

  1. const util = require('util'); 

然后用 promisify 方法將其轉(zhuǎn)換為 Promise:

  1. const fs = require('fs'); 
  2. const readFile = util.promisify(fs.readFile); 

現(xiàn)在,把新創(chuàng)建的函數(shù)用作 promise:

  1. readFile('./sample.txt', 'utf-8') 
  2.     .then(data => { 
  3.         console.log(data); 
  4.     }) 
  5.     .catch(err => { 
  6.         console.log(err); 
  7.     }); 

另外也可以用下面這個(gè)示例中給出的 async/await 關(guān)鍵字:

  1. const fs = require('fs'); 
  2. const util = require('util'); 
  3.  
  4. const readFile = util.promisify(fs.readFile); 
  5.  
  6. (async () => { 
  7.     try { 
  8.         const content = await readFile('./sample.txt', 'utf-8'); 
  9.         console.log(content); 
  10.     } catch (err) { 
  11.         console.error(err); 
  12.     } 
  13. })(); 

你只能在用 async 創(chuàng)建的函數(shù)中使用 await 關(guān)鍵字,這也是為什么要使用函數(shù)包裝器的原因。函數(shù)包裝器也被稱為立即調(diào)用的函數(shù)表達(dá)式。

如果你的回調(diào)不遵循這個(gè)特定標(biāo)準(zhǔn)也不用擔(dān)心。util.promisify() 函數(shù)可讓你自定義轉(zhuǎn)換是如何發(fā)生的。

注意:Promise 在被引入后不久就開始流行了。Node.js 已經(jīng)將大部分核心函數(shù)從回調(diào)轉(zhuǎn)換成了基于 Promise 的API。

如果需要用 Promise 處理文件,可以用 Node.js 附帶的庫(kù)(https://nodejs.org/docs/latest-v10.x/api/fs.html#fs_fs_promises_api)。

現(xiàn)在你已經(jīng)了解了如何將 Node.js 標(biāo)準(zhǔn)樣式回調(diào)隱含到 Promise 中。從 Node.js 8 開始,這個(gè)模塊僅在 Node.js 上可用。如果你用的是瀏覽器或早期版本版本的 Node,則最好創(chuàng)建自己的基于 Promise 的函數(shù)版本。

2. 創(chuàng)建你自己的 Promise

讓我們討論一下怎樣把回調(diào)轉(zhuǎn)為 util.promisify() 函數(shù)的 promise。

思路是創(chuàng)建一個(gè)新的包含回調(diào)函數(shù)的 Promise 對(duì)象。如果回調(diào)函數(shù)返回錯(cuò)誤,就拒絕帶有該錯(cuò)誤的Promise。如果回調(diào)函數(shù)返回非錯(cuò)誤輸出,就解決并輸出 Promise。

先把回調(diào)轉(zhuǎn)換為一個(gè)接受固定參數(shù)的函數(shù)的 promise 開始:

  1. const fs = require('fs'); 
  2.  
  3. const readFile = (fileName, encoding) => { 
  4.     return new Promise((resolve, reject) => { 
  5.         fs.readFile(fileName, encoding, (err, data) => { 
  6.             if (err) { 
  7.                 return reject(err); 
  8.             } 
  9.  
  10.             resolve(data); 
  11.         }); 
  12.     }); 
  13.  
  14. readFile('./sample.txt') 
  15.     .then(data => { 
  16.         console.log(data); 
  17.     }) 
  18.     .catch(err => { 
  19.         console.log(err); 
  20.     }); 

新函數(shù) readFile() 接受了用來讀取 fs.readFile() 文件的兩個(gè)參數(shù)。然后創(chuàng)建一個(gè)新的 Promise 對(duì)象,該對(duì)象包裝了該函數(shù),并接受回調(diào),在本例中為 fs.readFile()。

要 reject Promise 而不是返回錯(cuò)誤。所以代碼中沒有立即把數(shù)據(jù)輸出,而是先 resolve 了Promise。然后像以前一樣使用基于 Promise 的 readFile() 函數(shù)。

接下來看看接受動(dòng)態(tài)數(shù)量參數(shù)的函數(shù):

  1. const getMaxCustom = (callback, ...args) => { 
  2.     let max = -Infinity; 
  3.  
  4.     for (let i of args) { 
  5.         if (i > max) { 
  6.             max = i
  7.         } 
  8.     } 
  9.  
  10.     callback(max); 
  11.  
  12. getMaxCustom((max) => { console.log('Max is ' + max) }, 10, 2, 23, 1, 111, 20); 

第一個(gè)參數(shù)是 callback 參數(shù),這使它在接受回調(diào)的函數(shù)中有點(diǎn)與眾不同。

轉(zhuǎn)換為 promise 的方式和上一個(gè)例子一樣。創(chuàng)建一個(gè)新的 Promise 對(duì)象,這個(gè)對(duì)象包裝使用回調(diào)的函數(shù)。如果遇到錯(cuò)誤,就 reject,當(dāng)結(jié)果出現(xiàn)時(shí)將會(huì) resolve。

我們的 promise 版本如下:

  1. const getMaxPromise = (...args) => { 
  2.     return new Promise((resolve) => { 
  3.         getMaxCustom((max) => { 
  4.             resolve(max); 
  5.         }, ...args); 
  6.     }); 
  7.  
  8. getMaxCustom(10, 2, 23, 1, 111, 20) 
  9.     .then(max => console.log(max)); 

在創(chuàng)建 promise 時(shí),不管函數(shù)是以非標(biāo)準(zhǔn)方式還是帶有許多參數(shù)使用回調(diào)都無關(guān)緊要。我們可以完全控制它的完成方式,并且原理是一樣的。

總結(jié)

盡管現(xiàn)在回調(diào)已成為 JavaScript 中利用異步代碼的默認(rèn)方法,但 Promise 是一種更現(xiàn)代的方法,它更容易使用。如果遇到了使用回調(diào)的代碼庫(kù),那么現(xiàn)在就可以把它轉(zhuǎn)換為 Promise。

在本文中,我們首先學(xué)到了如何 在Node.js 中使用 utils.promisfy() 方法將接受回調(diào)的函數(shù)轉(zhuǎn)換為 Promise。然后,了解了如何創(chuàng)建自己的 Promise 對(duì)象,并在對(duì)象中包裝了無需使用外部庫(kù)即可接受回調(diào)的函數(shù)。這樣許多舊 JavaScript 代碼可以輕松地與現(xiàn)代的代碼庫(kù)和混合在一起。

 

責(zé)任編輯:趙寧寧 來源: 前端先鋒
相關(guān)推薦

2022-10-12 08:00:00

語音識(shí)別Node.js音頻質(zhì)量

2025-07-11 08:09:07

Node.jsAPI權(quán)限

2022-06-04 07:21:19

requireJSNode.js

2021-05-21 09:36:42

開發(fā)技能代碼

2016-08-11 14:02:02

NodeJS前端

2024-01-05 08:49:15

Node.js異步編程

2020-04-15 15:48:03

Node.jsstream前端

2013-11-01 09:34:56

Node.js技術(shù)

2015-03-10 10:59:18

Node.js開發(fā)指南基礎(chǔ)介紹

2017-03-20 13:43:51

Node.js內(nèi)存泄漏

2017-03-19 16:40:28

漏洞Node.js內(nèi)存泄漏

2013-12-13 09:42:30

Node.js技術(shù)

2021-12-25 22:29:57

Node.js 微任務(wù)處理事件循環(huán)

2020-05-29 15:33:28

Node.js框架JavaScript

2012-02-03 09:25:39

Node.js

2021-03-04 23:12:57

Node.js異步迭代器開發(fā)

2020-11-02 11:40:24

Node.jsRequire前端

2021-12-01 00:05:03

Js應(yīng)用Ebpf

2011-09-08 13:53:31

Node.js

2011-09-08 14:16:12

Node.js
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

久久久久久国产精品三级玉女聊斋 | 午夜精品亚洲一区二区三区嫩草| 99精品国产在热久久| 久久伊人资源站| 噜噜噜久久亚洲精品国产品小说| 九九九九精品九九九九| 国产网站在线播放| 91农村精品一区二区在线| 人人做人人爽| 成人午夜av| 欧美大片国产精品| 神马午夜伦理不卡| 一区二区三区精品国产| 成人国产视频在线| 亚洲精品成a人ⅴ香蕉片| 日韩欧美一区二区三区四区五区| 国产欧美一区二区精品忘忧草| 日本福利小视频| 欧美三日本三级三级在线播放| 美女91在线| 久久久亚洲综合网站| 91在线视频播放| 好吊日视频在线观看| 黑人欧美xxxx| 国产精品蜜芽在线观看| 久久久久久久综合狠狠综合| 亚洲精品91美女久久久久久久| 中文字幕日韩在线| av在线一区二区三区| 麻豆影视在线| 国产精品h片在线播放| 亚洲专区免费| 精品视频免费| 国产精一区二区三区| 日韩午夜影院| 3d动漫精品啪啪一区二区三区免费 | 欧美黑粗硬大| 亚洲国产精品二十页| 蜜桃一级网站.| 日韩激情第一页| 亚洲国产免费看| 国产精品久久一区二区三区不卡| 777777777亚洲妇女| 久久99久久久久| 国产经典第一页| 成人a在线视频| 亚洲免费观看高清在线观看| sm捆绑调教国产免费网站在线观看| 国产+人+亚洲| 欧美午夜宅男影院在线观看| 亚洲精品黄色| 在线观看黄av| 另类小说综合网| 亚洲天堂av女优| 国产欧美精品在线观看| 午夜电影福利| 久久久久久99| 一区二区三区无码高清视频| 中文字幕高清不卡| 蜜臀av免费一区二区三区| av一级在线| 亚洲无玛一区| 最近免费中文字幕视频2019| 欧美电影网址| 免费av网页| 亚洲精品二区| 成人国内精品久久久久一区| 日韩视频国产视频| 日本一区二区免费在线观看视频| 日韩国产在线| ririsao久久精品一区| 久久久99爱| 欧美激情一级精品国产| 在线观看成人免费视频| 另类成人小视频在线| 91嫩草精品| heyzo一区| 激情综合闲人网| 欧美黄网站在线观看| 久草精品电影| 国产精品免费久久久久久| 日韩欧美一区二区视频| 一区二区三区91| 久久久国产精华| 国内外成人在线| 国产午夜久久| 欧美综合另类| 91国产一区| 日韩激情电影| 亚洲视频国产精品| jizz亚洲大全| 91极品美女在线| 鲁大师成人一区二区三区| 两个人看的在线视频www| 成年人福利视频| 中文字幕日韩一区二区三区| 一区二区日本| 精品日韩av一区二区| 夜夜春成人影院| 国产特级淫片高清视频| 99久免费精品视频在线观看| 久久精品 人人爱| 91丨九色丨国产| 久久久久久成人精品| 91黄色小视频| 性久久久久久久久久久久| 91麻豆成人久久精品二区三区| 国产乱一区二区| 成人国产精品免费观看视频| 亚洲国产欧美国产综合一区| 欧美一二区在线观看| 97影院秋霞午夜在线观看| 伊人久久综合一区二区| 日韩av懂色| 国产精品99在线观看| 美女mm1313爽爽久久久蜜臀| 国产精品资源在线| 亚洲激情av在线| 欧美久久免费观看| 欧美日韩国产成人| 国产欧美日本在线| 男女污污的视频| 色在线视频网| 国产精品久久久久久久免费观看 | 人禽交欧美网站| 日韩午夜激情免费电影| 久久久久久久久久久久91| 亚洲一区免费看| 成人免费看视频| 看亚洲a级一级毛片| av大大超碰在线| 亚洲高清国产精品| 在线播放91| 污污的网站在线看| www.久久久.com| 秋霞午夜理伦电影在线观看| 亚洲精品久久久久久久蜜桃臀| 久久精品国产精品青草色艺| 97视频资源在线观看| 在线播放日韩精品| 中文字幕日韩av电影| 久久97久久97精品免视看| 久久天天躁狠狠躁夜夜av| 在线观看欧美日韩国产| 久久久久久尹人网香蕉| 国产精品一区二区三区在线观 | 69av成年福利视频| 四虎av网址| 日产精品99久久久久久| 国产一区自拍视频| 欧美日韩一级片网站| 久久精品一级爱片| 日韩av不卡一区| www国产精品av| av在线不卡观看免费观看| 999在线视频| 欧美aaaaaaaa| 亚洲免费资源| 亚洲91视频| 免费黄网站欧美| 欧美日韩精选| 97精品在线| 亚洲永久精品国产| 国产精品免费一区二区三区四区| 中文字幕欧美人妻精品一区| 香蕉大人久久国产成人av| 国产精品麻豆| 日韩影院精彩在线| 亚洲精品成人av| 经典三级在线视频| 电影在线高清| 精品不卡一区| 综合久久十次| 窝窝社区一区二区| 国产亚洲精品久久久久婷婷瑜伽| 国产精品亚洲视频| 欧美日韩国产123区| 精品中文字幕在线2019| 日韩av影视| 日本国产在线播放| 91极品在线| 色av手机在线| 亚洲美女少妇无套啪啪呻吟| 日韩精品国产欧美| 欧美韩国日本不卡| 最近2019好看的中文字幕免费| 久久99久久99精品中文字幕| 欧美极品少妇无套实战| 日韩经典av| 精品成人久久| 欧美日韩一区二区在线观看| 成人av片在线观看| 在线观看91久久久久久| 欧美亚洲另类久久综合| 国产乱子伦三级在线播放| 欧美一区二区三区久久精品| 欧美日韩国产精品一区| 久久精品日产第一区二区三区乱码| 麻豆视频在线观看免费网站| 91精品99|