NodeJS的Promise的用法解析
來源:易賢網(wǎng) 閱讀:1058 次 日期:2016-07-02 14:02:34
溫馨提示:易賢網(wǎng)小編為您整理了“NodeJS的Promise的用法解析”,方便廣大網(wǎng)友查閱!

下面小編就為大家?guī)硪黄狽odeJS的Promise的用法解析。小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考

Javascript的特點(diǎn)是異步,Javascript不能等待,如果你實(shí)現(xiàn)某件需要等待的事情,你不能停在那里一直等待結(jié)果回來,相反,底線是使用回調(diào)callback:你定義一個函數(shù),這個函數(shù)只有等到結(jié)果可用時才能被調(diào)用。

這種回調(diào)模型對于好的代碼組織是沒有問題的,但是也可以通過從原始回調(diào)切換到promise解決很多問題,將promise看成是一個標(biāo)準(zhǔn)的數(shù)據(jù)容器,這樣會簡化你的代碼組織,可以成為基于promise的架構(gòu)。

什么是Promise?

一個promise是一個帶有".then()"方法的對象,其代表的是一個操作的結(jié)果可能還沒有或不知道,無論誰訪問這個對象,都能夠使用".then()"方法加入回調(diào)等待操作出現(xiàn)成功結(jié)果或失敗時的提醒通知,。

那么為什么這樣做好處優(yōu)于回調(diào)呢?標(biāo)準(zhǔn)的回調(diào)模式在我們處理請求時需要同時提供回調(diào)函數(shù):

request(url, function(error, response) { 

 // handle success or error.

});

doSomethingElse(); 

很不幸,這段代碼意味著這個request函數(shù)并不知道它自己什么時候能夠完成,當(dāng)然也沒有必要,我們最終通過回調(diào)傳遞結(jié)果。這會導(dǎo)致多個回調(diào)形成了嵌套回調(diào),或者稱為回調(diào)陷阱。

queryTheDatabase(query, function(error, result) { 

 request(url, function(error, response) {

  doSomethingElse(response, function(error, result) {

   doAnotherThing(result, function(error, result) {

    request(anotherUrl, function(error, response) {

     ...

    });

   });

  });

 });

});

Promise能夠解決這種問題,允許低層代碼創(chuàng)建一個request然后返回一個對象,其代表著未完成的操作,讓調(diào)用者去決定應(yīng)該加入什么回調(diào)。

Promise是什么?

promise是一個異步編程的抽象,它是一個返回值或拋出exception的代理對象,一般promise對象都有一個then方法,這個then方法是我們?nèi)绾潍@得返回值(成功實(shí)現(xiàn)承諾的結(jié)果值,稱為fulfillment)或拋出exception(拒絕承諾的理由,稱為rejection),then是用兩個可選的回調(diào)作為參數(shù),我們可以稱為onFulfilled和OnRejected:

var promise = doSomethingAync()

promise.then(onFulfilled, onRejected)

當(dāng)這個promise被解決了,也就是異步過程完成后,onFulfilled和OnRejected中任何一個將被調(diào)用,

因此,一個promise有下面三個不同狀態(tài):

■pending待承諾 - promise初始狀態(tài)

■fulfilled實(shí)現(xiàn)承諾 - 一個承諾成功實(shí)現(xiàn)狀態(tài)

■rejected拒絕承諾 - 一個承諾失敗的狀態(tài)

以讀取文件為案例,下面是使用回調(diào)實(shí)現(xiàn)讀取文件后應(yīng)該做什么事情(輸出打印):

readFile(function (err, data) {

 if (err) return console.error(err)

 console.log(data)

})

如果我們的readFile函數(shù)返回一個promise,那么我們可以如下實(shí)現(xiàn)同樣的邏輯(輸出打印):

var promise = readFile()

promise.then(console.log, console.error)

這里我們有了一個值promise代表的是異步操作,我們能夠一直傳遞這個值promise,任何人訪問這個值都能夠使用then來消費(fèi)使用它,無論這個值代表的異步操作是否完成或沒有完成,我們也能保證異步的結(jié)果不會改變,因?yàn)檫@個promise代表的異步操作只會執(zhí)行一次,狀態(tài)是要么fulfilled要么是rejected。

理解Promise

Promise可能是不同于日常直覺,為了理解它,一些重要原理必須記牢: .then()總是返回一個新的promise.,如下面代碼:

var promise = readFile()

var promise2 = promise.then(readAnotherFile, console.error)

這里then的參數(shù)readAnotherFile, console.error是代表異步操作成功后的動作onFulfilled或失敗后的動作OnRejected,也就是說,讀取文件成功后執(zhí)行readAnotherFile函數(shù),否則失敗打印記錄錯誤。這種實(shí)現(xiàn)是兩個中只有一種可能。

我們再看下面上述代碼如下:

var promise = readFile()

var promise2 = promise.then(function (data) {

 return readAnotherFile() // 如果readFile成功,執(zhí)行readAnotherFile

}, function (err) {

 console.error(err) // 如果readFile不成功,記錄,但是還是執(zhí)行readAnotherFile

 return readAnotherFile()

})

promise2.then(console.log, console.error) // readAnotherFile函數(shù)的執(zhí)行結(jié)果

因?yàn)閠hen返回一個promise,它意味著promise能夠被chain串行鏈條花,這樣能避免回調(diào)地獄:

readFile()

 .then(readAnotherFile)

 .then(doSomethingElse)

 .then(...)

Promise法則有兩部分必須分離:

(1).then()總是返回一個新的promise,每次你調(diào)用它,它不管回調(diào)做什么,因?yàn)?then()在回調(diào)被調(diào)用之前已經(jīng)給了你一個承諾promise,回調(diào)的行為只影響承諾promise的實(shí)施,如果回調(diào)返回一個值,那么promise將使用那個值,如果這個值是一個promise,返回這個promise實(shí)施后的值給這個值,如果回調(diào)拋出錯誤,promise將拒絕錯誤。

(2)被.then()返回的promise是一個新的promise,它不同于那些.then()被調(diào)用的promise,promise長長的鏈條有時會好些隱藏這個事實(shí),不管如何,每次.then()調(diào)用都會產(chǎn)生一個新的promise,這里必須注意的是你真正需要考慮的是你最后調(diào)用.then()可能代表失敗,那么如果你不捕獲這種失敗,那么容易導(dǎo)致你的錯誤exception消失。

一些人認(rèn)為.then()串聯(lián)鏈條調(diào)用很類似fluent風(fēng)格,但是長長的promise鏈條會讓人迷惑,最后切分為一個個有意義的函數(shù):

function getTasks() { 

 return $http.get('http://example.com/api/v1/tasks')

  .then(function(response) {

   return response.data;

  });

}

function getMyTasks() { 

 return getTasks()

  .then(function(tasks) {

   return filterTasks(tasks, {

    owner: user.username

   });

  });

}

在這個例子中,兩個函數(shù)各自獲得一個promise,攜帶了一個回調(diào)函數(shù)。

有趣的Promise

同樣的promise能夠接受任何數(shù)目的回調(diào),當(dāng)一個Promise被解決實(shí)施后,其中所有回調(diào)函數(shù)都會被調(diào)用,此外,一個promise在被解決實(shí)施后,甚至可以接受一個新的回調(diào),這些回調(diào)完成能以正常方式被調(diào)用,這就允許我們使用回調(diào)實(shí)現(xiàn)簡單形式的緩存:

var tasksPromise; 

function getTasks() { 

 taskPromise = taskPromise || getTasksFromTheServer();

 return taskPromise;

}

這個案例中,getTasks()函數(shù)可以被任意次數(shù)調(diào)用,它總是返回銅牙的promise,其中函數(shù)getTasksFromTheServer()卻只是被調(diào)用一次。

以上這篇NodeJS的Promise的用法就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考

更多信息請查看網(wǎng)絡(luò)編程
易賢網(wǎng)手機(jī)網(wǎng)站地址:NodeJS的Promise的用法解析
由于各方面情況的不斷調(diào)整與變化,易賢網(wǎng)提供的所有考試信息和咨詢回復(fù)僅供參考,敬請考生以權(quán)威部門公布的正式信息和咨詢?yōu)闇?zhǔn)!

2025國考·省考課程試聽報名

  • 報班類型
  • 姓名
  • 手機(jī)號
  • 驗(yàn)證碼
關(guān)于我們 | 聯(lián)系我們 | 人才招聘 | 網(wǎng)站聲明 | 網(wǎng)站幫助 | 非正式的簡要咨詢 | 簡要咨詢須知 | 新媒體/短視頻平臺 | 手機(jī)站點(diǎn) | 投訴建議
工業(yè)和信息化部備案號:滇ICP備2023014141號-1 云南省教育廳備案號:云教ICP備0901021 滇公網(wǎng)安備53010202001879號 人力資源服務(wù)許可證:(云)人服證字(2023)第0102001523號
云南網(wǎng)警備案專用圖標(biāo)
聯(lián)系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關(guān)注公眾號:hfpxwx
咨詢QQ:1093837350(9:00—18:00)版權(quán)所有:易賢網(wǎng)
云南網(wǎng)警報警專用圖標(biāo)