NodeJS的Promise的用法解析
Javascript的特點(diǎn)是異步,Javascript不能等待,如果你實(shí)現(xiàn)某件需要等待的事情,你不能停在那里一直等待結(jié)果回來(lái),相反,底線是使用回調(diào)callback:你定義一個(gè)函數(shù),這個(gè)函數(shù)只有等到結(jié)果可用時(shí)才能被調(diào)用。
這種回調(diào)模型對(duì)于好的代碼組織是沒(méi)有問(wèn)題的,但是也可以通過(guò)從原始回調(diào)切換到promise解決很多問(wèn)題,將promise看成是一個(gè)標(biāo)準(zhǔn)的數(shù)據(jù)容器,這樣會(huì)簡(jiǎn)化你的代碼組織,可以成為基于promise的架構(gòu)。
什么是Promise?
一個(gè)promise是一個(gè)帶有".then()"方法的對(duì)象,其代表的是一個(gè)操作的結(jié)果可能還沒(méi)有或不知道,無(wú)論誰(shuí)訪問(wèn)這個(gè)對(duì)象,都能夠使用".then()"方法加入回調(diào)等待操作出現(xiàn)成功結(jié)果或失敗時(shí)的提醒通知,。
那么為什么這樣做好處優(yōu)于回調(diào)呢?標(biāo)準(zhǔn)的回調(diào)模式在我們處理請(qǐng)求時(shí)需要同時(shí)提供回調(diào)函數(shù):
request(url, function(error, response) { // handle success or error. }); doSomethingElse();
很不幸,這段代碼意味著這個(gè)request函數(shù)并不知道它自己什么時(shí)候能夠完成,當(dāng)然也沒(méi)有必要,我們最終通過(guò)回調(diào)傳遞結(jié)果。這會(huì)導(dǎo)致多個(gè)回調(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能夠解決這種問(wèn)題,允許低層代碼創(chuàng)建一個(gè)request然后返回一個(gè)對(duì)象,其代表著未完成的操作,讓調(diào)用者去決定應(yīng)該加入什么回調(diào)。
Promise是什么?
promise是一個(gè)異步編程的抽象,它是一個(gè)返回值或拋出exception的代理對(duì)象,一般promise對(duì)象都有一個(gè)then方法,這個(gè)then方法是我們?nèi)绾潍@得返回值(成功實(shí)現(xiàn)承諾的結(jié)果值,稱為fulfillment)或拋出exception(拒絕承諾的理由,稱為rejection),then是用兩個(gè)可選的回調(diào)作為參數(shù),我們可以稱為onFulfilled和OnRejected:
var promise = doSomethingAync()
promise.then(onFulfilled, onRejected)
當(dāng)這個(gè)promise被解決了,也就是異步過(guò)程完成后,onFulfilled和OnRejected中任何一個(gè)將被調(diào)用,
因此,一個(gè)promise有下面三個(gè)不同狀態(tài):
■pending待承諾 - promise初始狀態(tài)
■fulfilled實(shí)現(xiàn)承諾 - 一個(gè)承諾成功實(shí)現(xiàn)狀態(tài)
■rejected拒絕承諾 - 一個(gè)承諾失敗的狀態(tài)
以讀取文件為案例,下面是使用回調(diào)實(shí)現(xiàn)讀取文件后應(yīng)該做什么事情(輸出打印):
readFile(function (err, data) { if (err) return console.error(err) console.log(data) })
如果我們的readFile函數(shù)返回一個(gè)promise,那么我們可以如下實(shí)現(xiàn)同樣的邏輯(輸出打印):
var promise = readFile()
promise.then(console.log, console.error)
這里我們有了一個(gè)值promise代表的是異步操作,我們能夠一直傳遞這個(gè)值promise,任何人訪問(wèn)這個(gè)值都能夠使用then來(lái)消費(fèi)使用它,無(wú)論這個(gè)值代表的異步操作是否完成或沒(méi)有完成,我們也能保證異步的結(jié)果不會(huì)改變,因?yàn)檫@個(gè)promise代表的異步操作只會(huì)執(zhí)行一次,狀態(tài)是要么fulfilled要么是rejected。
理解Promise
Promise可能是不同于日常直覺(jué),為了理解它,一些重要原理必須記牢: .then()總是返回一個(gè)新的promise.,如下面代碼:
var promise = readFile()
var promise2 = promise.then(readAnotherFile, console.error)
這里then的參數(shù)readAnotherFile, console.error是代表異步操作成功后的動(dòng)作onFulfilled或失敗后的動(dòng)作OnRejected,也就是說(shuō),讀取文件成功后執(zhí)行readAnotherFile函數(shù),否則失敗打印記錄錯(cuò)誤。這種實(shí)現(xiàn)是兩個(gè)中只有一種可能。
我們?cè)倏聪旅嫔鲜龃a如下:
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返回一個(gè)promise,它意味著promise能夠被chain串行鏈條花,這樣能避免回調(diào)地獄:
readFile() .then(readAnotherFile) .then(doSomethingElse) .then(...)
Promise法則有兩部分必須分離:
(1).then()總是返回一個(gè)新的promise,每次你調(diào)用它,它不管回調(diào)做什么,因?yàn)?then()在回調(diào)被調(diào)用之前已經(jīng)給了你一個(gè)承諾promise,回調(diào)的行為只影響承諾promise的實(shí)施,如果回調(diào)返回一個(gè)值,那么promise將使用那個(gè)值,如果這個(gè)值是一個(gè)promise,返回這個(gè)promise實(shí)施后的值給這個(gè)值,如果回調(diào)拋出錯(cuò)誤,promise將拒絕錯(cuò)誤。
(2)被.then()返回的promise是一個(gè)新的promise,它不同于那些.then()被調(diào)用的promise,promise長(zhǎng)長(zhǎng)的鏈條有時(shí)會(huì)好些隱藏這個(gè)事實(shí),不管如何,每次.then()調(diào)用都會(huì)產(chǎn)生一個(gè)新的promise,這里必須注意的是你真正需要考慮的是你最后調(diào)用.then()可能代表失敗,那么如果你不捕獲這種失敗,那么容易導(dǎo)致你的錯(cuò)誤exception消失。
一些人認(rèn)為.then()串聯(lián)鏈條調(diào)用很類似fluent風(fēng)格,但是長(zhǎng)長(zhǎng)的promise鏈條會(huì)讓人迷惑,最后切分為一個(gè)個(gè)有意義的函數(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 }); }); }
在這個(gè)例子中,兩個(gè)函數(shù)各自獲得一個(gè)promise,攜帶了一個(gè)回調(diào)函數(shù)。
有趣的Promise
同樣的promise能夠接受任何數(shù)目的回調(diào),當(dāng)一個(gè)Promise被解決實(shí)施后,其中所有回調(diào)函數(shù)都會(huì)被調(diào)用,此外,一個(gè)promise在被解決實(shí)施后,甚至可以接受一個(gè)新的回調(diào),這些回調(diào)完成能以正常方式被調(diào)用,這就允許我們使用回調(diào)實(shí)現(xiàn)簡(jiǎn)單形式的緩存:
var tasksPromise; function getTasks() { taskPromise = taskPromise || getTasksFromTheServer(); return taskPromise; }
這個(gè)案例中,getTasks()函數(shù)可以被任意次數(shù)調(diào)用,它總是返回銅牙的promise,其中函數(shù)getTasksFromTheServer()卻只是被調(diào)用一次。
以上這篇NodeJS的Promise的用法就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- node使用promise替代回調(diào)函數(shù)
- async/await與promise(nodejs中的異步操作問(wèn)題)
- NodeJS中利用Promise來(lái)封裝異步函數(shù)
- nodejs中簡(jiǎn)單實(shí)現(xiàn)Javascript Promise機(jī)制的實(shí)例
- node.js中使用q.js實(shí)現(xiàn)api的promise化
- 基于promise.js實(shí)現(xiàn)nodejs的promises庫(kù)
- Nodejs學(xué)習(xí)筆記之Global Objects全局對(duì)象
- 用nodejs訪問(wèn)ActiveX對(duì)象,以操作Access數(shù)據(jù)庫(kù)為例。
- 詳解nodeJS之二進(jìn)制buffer對(duì)象
- Node.js 基礎(chǔ)教程之全局對(duì)象
- 深入理解Node內(nèi)建模塊和對(duì)象
- node.js Promise對(duì)象的使用方法實(shí)例分析
相關(guān)文章
javascript過(guò)濾數(shù)組重復(fù)元素的實(shí)現(xiàn)方法
這篇文章主要介紹了javascript過(guò)濾數(shù)組重復(fù)元素的實(shí)現(xiàn)方法的相關(guān)資料,需要的朋友可以參考下2017-05-05使用eslint和githooks統(tǒng)一前端風(fēng)格的技巧
這篇文章主要介紹了使用eslint和githooks統(tǒng)一前端風(fēng)格,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07JS實(shí)現(xiàn)Tab欄切換的兩種方式案例詳解
這篇文章主要介紹了JS實(shí)現(xiàn)Tab欄切換的兩種方式,一種是面向過(guò)程的寫法,一種是面向?qū)ο蟮膶懛ǎ疚慕o大家分享詳細(xì)案例代碼,需要的朋友可以參考下2022-08-08基于javascript實(shí)現(xiàn)最簡(jiǎn)單選項(xiàng)卡切換
這篇文章主要為大家詳細(xì)介紹了基于javascript實(shí)現(xiàn)最簡(jiǎn)單選項(xiàng)卡切換,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-02-02javascript中slice(),splice(),split(),substring(),substr()使用方法
這篇文章主要介紹了javascript中slice(),splice(),split(),substring(),substr()使用方法,需要的朋友可以參考下2015-03-03