2021-03-17 分類(lèi): 網(wǎng)站建設(shè)
Node的產(chǎn)生,大大推動(dòng)了Javascript
這門(mén)語(yǔ)言在服務(wù)端的發(fā)展,使得前端人員可以以很低的門(mén)檻轉(zhuǎn)向后端開(kāi)發(fā)。
當(dāng)然,這并不代表迸發(fā)成了全棧。全棧的技能很集中,絕不僅僅是前端會(huì)寫(xiě)一些HTML
和一些交互,后臺(tái)熟悉數(shù)據(jù)庫(kù)的增刪查改。
想必接觸過(guò)Node的人都知道,Node是以異步(Async)回調(diào)著稱(chēng)的,其異步性提高了程序的執(zhí)行效率,但同時(shí)也減少了程序的可讀性。如果我們有幾個(gè)異步操作,并且后一個(gè)操作需要前一個(gè)操作返回的數(shù)據(jù)才能執(zhí)行,這樣按照Node的一般執(zhí)行規(guī)律,要實(shí)現(xiàn)有序的異步操作,通常是一層加一層嵌套下去。
為了解決這個(gè)問(wèn)題,ES6提出了Promise
的實(shí)現(xiàn)。
Promise 對(duì)象用于一個(gè)異步操作的最終完成(或失?。┘捌浣Y(jié)果值的表示。簡(jiǎn)單點(diǎn)說(shuō),它就是用于處理異步操作的,異步處理成功了就執(zhí)行成功的操作,異步處理失敗了就捕獲錯(cuò)誤或者停止后續(xù)操作。
它的一般表示形式為:
new Promise(
/* executor */
function(resolve, reject) {
if (/* success */) {
// ...執(zhí)行代碼
resolve();
} else { /* fail */
// ...執(zhí)行代碼
reject();
}
}
);
其中,Promise中的參數(shù)executor
是一個(gè)執(zhí)行器函數(shù),它有兩個(gè)參數(shù)resolve
和reject
。它內(nèi)部通常有一些異步操作,如果異步操作成功,則可以調(diào)用resolve()來(lái)將該實(shí)例的狀態(tài)置為fulfilled
,即已完成的,如果一旦失敗,可以調(diào)用reject()來(lái)將該實(shí)例的狀態(tài)置為rejected
,即失敗的。
我們可以把Promise對(duì)象看成是一條工廠(chǎng)的流水線(xiàn),對(duì)于流水線(xiàn)來(lái)說(shuō),從它的工作職能上看,它只有三種狀態(tài),一個(gè)是初始狀態(tài)(剛開(kāi)機(jī)的時(shí)候),一個(gè)是加工產(chǎn)品成功,一個(gè)是加工產(chǎn)品失?。ǔ霈F(xiàn)了某些故障)。同樣對(duì)于Promise對(duì)象來(lái)說(shuō),它也有三種狀態(tài):
pending
fulfilled
rejected
它只有兩種狀態(tài)可以轉(zhuǎn)化,即
并且這個(gè)狀態(tài)轉(zhuǎn)化是單向的,不可逆轉(zhuǎn),已經(jīng)確定的狀態(tài)(fulfilled/rejected)無(wú)法轉(zhuǎn)回初始狀態(tài)(pending)。
Promise對(duì)象含有then方法,then()調(diào)用后返回一個(gè)Promise對(duì)象,意味著實(shí)例化后的Promise對(duì)象可以進(jìn)行鏈?zhǔn)秸{(diào)用,而且這個(gè)then()方法可以接收兩個(gè)函數(shù),一個(gè)是處理成功后的函數(shù),一個(gè)是處理錯(cuò)誤結(jié)果的函數(shù)。
如下:
var promise1 = new Promise(function(resolve, reject) {
// 2秒后置為接收狀態(tài)
setTimeout(function() {
resolve('success');
}, 2000);
});
promise1.then(function(data) {
console.log(data); // success
}, function(err) {
console.log(err); // 不執(zhí)行
}).then(function(data) {
// 上一步的then()方法沒(méi)有返回值
console.log('鏈?zhǔn)秸{(diào)用:' + data); // 鏈?zhǔn)秸{(diào)用:undefined
}).then(function(data) {
// ....
});
在這里我們主要關(guān)注promise1.then()方法調(diào)用后返回的Promise對(duì)象的狀態(tài),是pending
還是fulfilled
,或者是rejected
?
返回的這個(gè)Promise對(duì)象的狀態(tài)主要是根據(jù)promise1.then()方法返回的值,大致分為以下幾種情況:
轉(zhuǎn)換實(shí)例如下:
var promise2 = new Promise(function(resolve, reject) {
// 2秒后置為接收狀態(tài)
setTimeout(function() {
resolve('success');
}, 2000);
});
promise2
.then(function(data) {
// 上一個(gè)then()調(diào)用了resolve,置為fulfilled態(tài)
console.log('第一個(gè)then');
console.log(data);
return '2';
})
.then(function(data) {
// 此時(shí)這里的狀態(tài)也是fulfilled, 因?yàn)樯弦徊椒祷亓?
console.log('第二個(gè)then');
console.log(data); // 2
return new Promise(function(resolve, reject) {
reject('把狀態(tài)置為rejected error'); // 返回一個(gè)rejected的Promise實(shí)例
});
}, function(err) {
// error
})
.then(function(data) {
/* 這里不運(yùn)行 */
console.log('第三個(gè)then');
console.log(data);
// ....
}, function(err) {
// error回調(diào)
// 此時(shí)這里的狀態(tài)也是fulfilled, 因?yàn)樯弦徊绞褂昧藃eject()來(lái)返回值
console.log('出錯(cuò):' + err); // 出錯(cuò):把狀態(tài)置為rejected error
})
.then(function(data) {
// 沒(méi)有明確指定返回值,默認(rèn)返回fulfilled
console.log('這里是fulfilled態(tài)');
});
catch()方法和then()方法一樣,都會(huì)返回一個(gè)新的Promise對(duì)象,它主要用于捕獲異步操作時(shí)出現(xiàn)的異常。因此,我們通常省略then()方法的第二個(gè)參數(shù),把錯(cuò)誤處理控制權(quán)轉(zhuǎn)交給其后面的catch()函數(shù),如下:
var promise3 = new Promise(function(resolve, reject) {
setTimeout(function() {
reject('reject');
}, 2000);
});
promise3.then(function(data) {
console.log('這里是fulfilled狀態(tài)'); // 這里不會(huì)觸發(fā)
// ...
}).catch(function(err) {
// 最后的catch()方法可以捕獲在這一條Promise鏈上的異常
console.log('出錯(cuò):' + err); // 出錯(cuò):reject
});
Promise.all()接收一個(gè)參數(shù),它必須是可以迭代的,比如數(shù)組。
它通常用來(lái)處理一些并發(fā)的異步操作,即它們的結(jié)果互不干擾,但是又需要異步執(zhí)行。它最終只有兩種狀態(tài):成功或者失敗。
它的狀態(tài)受參數(shù)內(nèi)各個(gè)值的狀態(tài)影響,即里面狀態(tài)全部為fulfilled
時(shí),它才會(huì)變成fulfilled
,否則變成rejected
。
成功調(diào)用后返回一個(gè)數(shù)組,數(shù)組的值是有序的,即按照傳入?yún)?shù)的數(shù)組的值操作后返回的結(jié)果。如下:
// 置為fulfilled狀態(tài)的情況
var arr = [1, 2, 3];
var promises = arr.map(function(e) {
return new Promise(function(resolve, reject) {
resolve(e * 5);
});
});
Promise.all(promises).then(function(data) {
// 有序輸出
console.log(data); // [5, 10, 15]
console.log(arr); // [1, 2, 3]
});
// 置為rejected狀態(tài)的情況
var arr = [1, 2, 3];
var promises2 = arr.map(function(e) {
return new Promise(function(resolve, reject) {
if (e === 3) {
reject('rejected');
}
resolve(e * 5);
});
});
Promise.all(promises2).then(function(data) {
// 這里不會(huì)執(zhí)行
console.log(data);
console.log(arr);
}).catch(function(err) {
console.log(err); // rejected
});
Promise.race()和Promise.all()類(lèi)似,都接收一個(gè)可以迭代的參數(shù),但是不同之處是Promise.race()的狀態(tài)變化不是全部受參數(shù)內(nèi)的狀態(tài)影響,一旦參數(shù)內(nèi)有一個(gè)值的狀態(tài)發(fā)生的改變,那么該P(yáng)romise的狀態(tài)就是改變的狀態(tài)。就跟race
單詞的字面意思一樣,誰(shuí)跑的快誰(shuí)贏。如下:
var p1 = new Promise(function(resolve, reject) {
setTimeout(resolve, 300, 'p1 doned');
});
var p2 = new Promise(function(resolve, reject) {
setTimeout(resolve, 50, 'p2 doned');
});
var p3 = new Promise(function(resolve, reject) {
setTimeout(reject, 100, 'p3 rejected');
});
Promise.race([p1, p2, p3]).then(function(data) {
// 顯然p2更快,所以狀態(tài)變成了fulfilled
// 如果p3更快,那么狀態(tài)就會(huì)變成rejected
console.log(data); // p2 doned
}).catch(function(err) {
console.log(err); // 不執(zhí)行
});
Promise.resolve()接受一個(gè)參數(shù)值,可以是普通的值
,具有then()方法的對(duì)象
和Promise實(shí)例
。正常情況下,它返回一個(gè)Promise對(duì)象,狀態(tài)為fulfilled
。但是,當(dāng)解析時(shí)發(fā)生錯(cuò)誤時(shí),返回的Promise對(duì)象將會(huì)置為rejected
態(tài)。如下:
// 參數(shù)為普通值
var p4 = Promise.resolve(5);
p4.then(function(data) {
console.log(data); // 5
});
// 參數(shù)為含有then()方法的對(duì)象
var obj = {
then: function() {
console.log('obj 里面的then()方法');
}
};
var p5 = Promise.resolve(obj);
p5.then(function(data) {
// 這里的值時(shí)obj方法里面返回的值
console.log(data); // obj 里面的then()方法
});
// 參數(shù)為Promise實(shí)例
var p6 = Promise.resolve(7);
var p7 = Promise.resolve(p6);
p7.then(function(data) {
// 這里的值時(shí)Promise實(shí)例返回的值
console.log(data); // 7
});
// 參數(shù)為Promise實(shí)例,但參數(shù)是rejected態(tài)
var p8 = Promise.reject(8);
var p9 = Promise.resolve(p8);
p9.then(function(data) {
// 這里的值時(shí)Promise實(shí)例返回的值
console.log('fulfilled:'+ data); // 不執(zhí)行
}).catch(function(err) {
console.log('rejected:' + err); // rejected: 8
});
Promise.reject()和Promise.resolve()正好相反,它接收一個(gè)參數(shù)值reason
,即發(fā)生異常的原因。此時(shí)返回的Promise對(duì)象將會(huì)置為rejected
態(tài)。如下:
var p10 = Promise.reject('手動(dòng)拒絕');
p10.then(function(data) {
console.log(data); // 這里不會(huì)執(zhí)行,因?yàn)槭莚ejected態(tài)
}).catch(function(err) {
console.log(err); // 手動(dòng)拒絕
}).then(function(data) {
// 不受上一級(jí)影響
console.log('狀態(tài):fulfilled'); // 狀態(tài):fulfilled
});
總之,除非Promise.then()方法內(nèi)部拋出異?;蛘呤敲鞔_置為rejected態(tài),否則它返回的Promise的狀態(tài)都是fulfilled態(tài),即完成態(tài),并且它的狀態(tài)不受它的上一級(jí)的狀態(tài)的影響。
大概常用的方法就寫(xiě)那么多,剩下的看自己實(shí)際需要再去了解。
解決Node回調(diào)地獄的不止有
Promise
,還有總結(jié)一些Promise常用的用法
當(dāng)前地址:http://aaarwkj.com/news29/105329.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站營(yíng)銷(xiāo)、用戶(hù)體驗(yàn)、定制網(wǎng)站、建站公司、微信公眾號(hào)、網(wǎng)站設(shè)計(jì)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶(hù)投稿、用戶(hù)轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話(huà):028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源:
創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容