為了解決回調(diào)地獄的問題,ES6(ECMAScript 2015)中新增了 Promise 的概念,Promise 是一個(gè)構(gòu)造函數(shù),可以用來創(chuàng)建 Promise 的實(shí)例 const p = new Promise(),new 出來的 Promise 實(shí)例對(duì)象,代表一個(gè)異步操作。
Promise.prototype 上包含一個(gè) .then() 方法, 每一次 new Promise() 構(gòu)造函數(shù)得到的實(shí)例對(duì)象,都可以通過原型鏈的方式訪問到 .then() 方法,例如 p.then()。
同時(shí).then() 方法用來預(yù)先指定成功和失敗的回調(diào)函數(shù)
p.then(成功的回調(diào)函數(shù),失敗的回調(diào)函數(shù))
p.then(result => { }, error => { })
調(diào)用 .then() 方法時(shí),成功的回調(diào)函數(shù)是必選的、失敗的回調(diào)函數(shù)是可選的?;谝陨系睦碚?,我們來看Promise的常見用法。
基于回調(diào)函數(shù)按順序讀取文件內(nèi)容,具體代碼如下:
//讀取文件1.txt
fs.readFile('./files/1.txt', 'utf8', (err1, r1) => {
if (err1) return console.log(err1.message) //讀取文件1失取
console, log(r1) //讀取文件 1 成功
//讀取文件2.txt
fs.readFile('./files/2.txt', 'utf8', (err2, r2) => { //詩(shī)取文件 2 失敗
if (err2) return console.log(err2.message)
console.log(r2) //讀取文件 2 成功
//讀取文件3.txt
fs.readFile('./files/3.txt', 'utf8', (err3, r3) => {
if (err3) return console.log(err3.message) //讀取文件 3失敗
console.log(r3) //讀取文件 3 成功
})
})
})
基于 then-fs 讀取文件內(nèi)容
由于 node.js 官方提供的 fs 模塊僅支持以回調(diào)函數(shù)的方式讀取文件,不支持 Promise 的調(diào)用方式。因此,需要先運(yùn)行如下的命令,安裝 then-fs 這個(gè)第三方包,從而支持我們基于 Promise 的方式讀取文件的內(nèi)容。
npm install then-fs
調(diào)用then-fs提供的readFile()方法,可以異步地讀取文件的內(nèi)容,C的返回值是Promise的實(shí)例對(duì)象。因此可以調(diào)用.then()方法為每個(gè)Promise異步操作指定成功和失敗之后的回調(diào)函數(shù)。示例代碼如下:
then-fs 的基本使用
調(diào)用 then-fs 提供的 readFile() 方法,可以異步地讀取文件的內(nèi)容,它的返回值是 Promise 的實(shí)例對(duì)象。因此可以調(diào)用 .then() 方法為每個(gè) Promise 異步操作指定成功和失敗之后的回調(diào)函數(shù)。示例代碼如下:
/★★
*基于Promise的方式讀取文件
*/
impoik: thenFs from 'then-fs'
//注意:.then()中的失敗回調(diào)是可選的,可以被省略
thenFs.readFile('./files/1.txt','utf8').then(r1=>{console.log(r1)},err1 =>{console.log(err1.message)})
thenFs.readFile('./files/2.txt','utf8').then(r2 =>{console.log(r2)},err2 =>{console.log(err2.message)})
thenFs.readFile('./files/3.txt','utf8').then(r3 =>{ console.log(r3)},err3 =>{console.log(err3.message)})
注意:上述的代碼無(wú)法保證文件的讀取順序,需要做進(jìn)一步的改進(jìn)!
.then() 方法的特性
如果上一個(gè) .then() 方法中返回了一個(gè)新的 Promise 實(shí)例對(duì)象,則可以通過下一個(gè) .then() 繼續(xù)進(jìn)行處理。通過 .then() 方法的鏈?zhǔn)秸{(diào)用,就解決了回調(diào)地獄的問題。
基于 Promise 按順序讀取文件的內(nèi)容
Promise 支持鏈?zhǔn)秸{(diào)用,從而來解決回調(diào)地獄的問題。示例代碼如下:
thenFs.readFile('./files/1.txt','utf8')// 1.返回值是Promise的實(shí)例對(duì)象
.then((r1)=>{// 2.通過.then為第一個(gè)Promise實(shí)例指定成功之后的回調(diào)函數(shù)
console.log(r1)
return thenFs.readFile('./files/2.txt',/utf8')// 3.在第一個(gè).then中返回一個(gè)新的Promise實(shí)例對(duì)象
})
.then((r2)=>{//4.繼續(xù)調(diào)用.then,為上一個(gè).then 的返回值(新的Promise 實(shí)例)指定成功之后的回調(diào)函數(shù)
console.log(r2)
return thenFs.readFile('./files/3.txt','utf8')// 5.在第二個(gè).then中再返回一個(gè)新的Promise實(shí)例對(duì)象
})
.then((r3)=>{// 6.繼續(xù)調(diào)用.then,為上一個(gè).then 的返回值(新的Promise實(shí)例)指定成功之后的回調(diào)函數(shù)
console.log(r3)
})
通過 .catch 捕獲錯(cuò)誤
在 Promise 的鏈?zhǔn)讲僮髦腥绻l(fā)生了錯(cuò)誤,可以使用 Promise.prototype.catch 方法進(jìn)行捕獲和處理:
thenFs.readFile('./files/11.txt','utf8')//文件不存在導(dǎo)致讀取失敗,后面的3個(gè).then 都不執(zhí)行
.then(r1 =>{
console.log(r1)
return thenFs.readFile('./files/2.txt','utf8')
})
.then(r2=> {
console.log(r2)
return thenFs.readFile('./files/3.txt', 'utf8')
})
.then(r3 =>(
console.log(r3)
})
.catch(err =>{//捕獲第1行發(fā)生的錯(cuò)誤,并輸出錯(cuò)誤的消息
console.log(err.message)
})
Promise.all() 方法
Promise.all() 方法會(huì)發(fā)起并行的 Promise 異步操作,等所有的異步操作全部結(jié)束后才會(huì)執(zhí)行下一步的 .then操作(等待機(jī)制)。示例代碼如下:
//1.定義一個(gè)數(shù)組,存放3個(gè)讀文件的異步操作
const promiseArr = [
thets.readFile('./files/11.txt','utf8').
thenFs.readFile('./files/2.txt','utf8'),
thenFs.readFile('./files/3.txt','utf8'),
]
// 2.將 Promise的數(shù)組,作為Promise.all()的參數(shù)
Promise.all(promiseArr)
.then(([r1,r2,r3])=>{//2.1所有文件讀取成功(等待機(jī)制)
console.log(r1, r2, r3)
})
.catch(err =>{// 2.2捕獲 Promise異步操作中的錯(cuò)誤
console.log(err.message)
})
注意:數(shù)組中 Promise 實(shí)例的順序,就是最終結(jié)果的順序!
Promise.race() 方法
Promise.race() 方法會(huì)發(fā)起并行的 Promise 異步操作,只要任何一個(gè)異步操作完成,就立即執(zhí)行下一步的 .then 操作(賽跑機(jī)制)。示例代碼如下:
//1.定義一個(gè)數(shù)組,存放3個(gè)讀文件的異步操作
const promiseArr =[
thenFs.readFile('./files/1.txt', 'utf8'),
thenFs.readFile('./files/2.txt', 'utf8').
thenFs.readFile('./files/3.txt', 'utf8').
]
// 2.將 Promise 的數(shù)組,作為 Promise.race()的參數(shù)
Pronise.race(promiseArr)
.then((result)=>{//2.1只要任何一個(gè)異步操作完成,就立即執(zhí)行成功的回調(diào)函數(shù)(賽跑機(jī)制)
console.log(result)
})
.catch(err =>{// 2.2捕獲Promise 異步操作中的錯(cuò)誤
console.log(err.message)
})