Javascript異步編程模型Promise模式詳細(xì)介紹
Promise 編程模式也被稱為 thenable,可以理解為 延遲后執(zhí)行。每個(gè) Promise 都擁有一個(gè)叫做 then 的唯一接口,當(dāng) Promise 失敗或成功時(shí),它就會(huì)進(jìn)行回調(diào)。它代表了一種可能會(huì)長(zhǎng)時(shí)間運(yùn)行而且不一定必須完成的操作結(jié)果。這種模式不會(huì)阻塞和等待長(zhǎng)時(shí)間的操作完成,而是返回一個(gè)代表了承諾的(promised)結(jié)果的對(duì)象。
當(dāng)前的許多 JavaScript 庫(kù)(如 jQuery 和 Dojo、AngularJS)均添加了這種稱為 Promise 的抽象。通過(guò)這些庫(kù),開發(fā)人員能夠在實(shí)際編程中使用 Promise 模式。
下面我們將以 jQuery 為例討論 JavaScript 庫(kù)是如何使用 Promise 模式的來(lái)處理異步的,其實(shí)就是通過(guò)回調(diào)的方式提供容錯(cuò)支持。在某個(gè)操作成功或失敗或任何情況下都執(zhí)行對(duì)應(yīng)的回調(diào),盡量把某段邏輯可能出現(xiàn)的情況都 handle 住。
首先讓我們來(lái)看看 jQuery 一般是如何操作的:
var $info = $("#info");
$.ajax({
url:"/echo/json/",
data: { json: JSON.stringify({"name": "someValue"}) },
type:"POST",
success: function(response)
{
$info.text(response.name);
}
});
在這個(gè)例子中,你可以看到當(dāng)設(shè)置成功后會(huì)指定一個(gè)回調(diào),是一種很好的回調(diào)方式,這并不是 Promise,jQuery 官方文檔也不再推薦這種方式(http://api.jquery.com/jQuery.ajax/#jqXHR)。 當(dāng) Ajax 調(diào)用完成后,它便會(huì)執(zhí)行 success 函數(shù)。根據(jù)庫(kù)所使用的異步操作,你可以使用各種不同的回調(diào)(即任務(wù)是否成功,都會(huì)進(jìn)行回調(diào),做出響應(yīng))。使用 Promise 模式會(huì)簡(jiǎn)化這個(gè)過(guò)程,異步操作只需返回一個(gè)對(duì)象調(diào)用。這個(gè) Promise 允許你調(diào)用一個(gè)叫做 then 的方法,然后讓你指定回調(diào)的 function(s) 個(gè)數(shù)。
下面讓我們來(lái)看看 jQuery 是如何建立 Promise 的:
var $info = $("#info");
$.ajax({
url: "/echo/json/",
data: {
json: JSON.stringify({
"name": "someValue"
})
},
type: "POST"
})
.then(function (response) {
$info.text(response.name);
});
jQuery ajax 對(duì)象通過(guò)返回 xhr 對(duì)象實(shí)現(xiàn) Promise 模式,所以我們可以調(diào)用 then 方法,這樣做的優(yōu)勢(shì)是你可以鏈?zhǔn)秸{(diào)用,實(shí)現(xiàn)獨(dú)立操作,如下所示 :
var $info = $("#info");
$.ajax({
url: "/echo/json/",
data: {
json: JSON.stringify({
"name": "someValue"
})
},
type: "POST"
})
.then(function (response) {
$info.text(response.name);
})
.then(function () {
$info.append("...More");
})
.done(function () {
$info.append("...finally!");
});
由于許多庫(kù)都開始采用 Promise 模式,所以異步操作會(huì)變的非常容易。但如果站在相反的角度思考,Promise 將會(huì)是什么樣子的呢?其中一個(gè)非常重要的模式是函數(shù)可以接受兩種功能,一個(gè)是成功時(shí)的回調(diào),另一個(gè)是失敗時(shí)的回調(diào)。
var $info = $("#info");
$.ajax({
// Change URL to see error happen
url: "/echo/json/",
data: {
json: JSON.stringify({
"name": "someValue"
})
},
type: "POST"
})
.then(function (response) {
// success
$info.text(response.name);
},
function () {
// failure
$info.text("bad things happen to good developers");
})
.always(function () {
$info.append("...finally");
});
需要注意的是,在 jQuery 里,無(wú)論成功還是失敗,我們都會(huì)使用一個(gè)調(diào)用來(lái)指定我們想要調(diào)用的。
其實(shí)這里也可以這樣來(lái)寫,這也是 jQuery 官方文檔里推薦的方法:
var $info = $("#info");
$.ajax({
// Change URL to see error happen
url: "/echo/json/",
data: {
json: JSON.stringify({
"name": "someValue"
})
},
type: "POST"
})
.done(function (response) {
// success
$info.text(response.name);
}).fail(function () {
// failure
$info.text("bad things happen to good developers");
})
.always(function () {
$info.append("...finally");
});
下面再來(lái)看看 AngularJS 是如何使用 Promise 模式的:
var m = angular.module("myApp", []);
m.factory("dataService", function ($q) {
function _callMe() {
var d = $q.defer();
setTimeout(function () {
d.resolve();
//defer.reject();
}, 100);
return d.promise;
}
return {
callMe: _callMe
};
});
function myCtrl($scope, dataService) {
$scope.name = "None";
$scope.isBusy = true;
dataService.callMe()
.then(function () {
// Successful
$scope.name = "success";
},
function () {
// failure
$scope.name = "failure";
})
.then(function () {
// Like a Finally Clause
$scope.isBusy = false;
});
}
你可以在 JSFiddle 里試試這些例子,并且看看會(huì)產(chǎn)生哪些效果。使用 Promise 來(lái)操作異步是一種非常簡(jiǎn)單的方式,而且還可以簡(jiǎn)化你的代碼,確實(shí)是一舉兩得的好方法。
更多關(guān)于Promise的介紹及示例,可以前往官網(wǎng)(http://www.promisejs.org/)查看。
- Javascript異步編程之你真的懂Promise嗎
- JavaScript異步編程之Promise的初步使用詳解
- 深入理解JS異步編程-Promise
- 詳解JavaScript異步編程中jQuery的promise對(duì)象的作用
- javascript使用Promise對(duì)象實(shí)現(xiàn)異步編程
- 異步JavaScript編程中的Promise使用方法
- javascript異步編程代碼書寫規(guī)范Promise學(xué)習(xí)筆記
- Javascript中的異步編程規(guī)范Promises/A詳細(xì)介紹
- JavaScript異步編程Promise模式的6個(gè)特性
- JS異步編程Promise對(duì)象詳解
相關(guān)文章
聊聊JS動(dòng)畫庫(kù) Velocity.js的使用
本篇文章主要介紹了聊聊JS動(dòng)畫庫(kù) Velocity.js的使用,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03淺談webpack 構(gòu)建性能優(yōu)化策略小結(jié)
webpack以其豐富的功能和靈活的配置而深受業(yè)內(nèi)吹捧,逐步取代了grunt和gulp成為大多數(shù)前端工程實(shí)踐中的首選,這篇文章主要介紹了淺談webpack 構(gòu)建性能優(yōu)化策略小結(jié),感興趣的小伙伴們可以參考一下2018-06-06js下關(guān)于onmouseout、事件冒泡的問(wèn)題經(jīng)驗(yàn)小結(jié)
第3次遇到這個(gè)問(wèn)題,于是總結(jié)了一下,將此短文發(fā)在首頁(yè),希望對(duì)瀏覽器事件機(jī)制有所了解的大俠們給予解答2010-12-12原生js實(shí)現(xiàn)給指定元素的后面追加內(nèi)容
給指定元素的后面追加一些自己想要實(shí)現(xiàn)的效果或者是一些屬性之類的,有木有遇到過(guò),接下來(lái)為大家詳細(xì)介紹下追加的實(shí)現(xiàn)代碼,感興趣的朋友可以參考下哈2013-04-04js獲取系統(tǒng)的根路徑實(shí)現(xiàn)介紹
js如何獲取系統(tǒng)的根路徑,在本文給出了詳細(xì)的方法,感興趣的朋友可以參考下,希望對(duì)大家有所幫助2013-09-09開發(fā)跨瀏覽器javascript常見注意事項(xiàng)
對(duì)于javascript的開發(fā)人員來(lái)說(shuō),多瀏覽器的支持性,一直是個(gè)問(wèn)題,每次都要經(jīng)過(guò)測(cè)試,多個(gè)瀏覽器,才能使用下面一些常見的一些注意事項(xiàng)。2009-01-01JavaScript獲取一個(gè)范圍內(nèi)日期的方法
這篇文章主要介紹了JavaScript獲取一個(gè)范圍內(nèi)日期的方法,涉及javascript操作日期的相關(guān)技巧,需要的朋友可以參考下2015-04-04JavaScript 實(shí)現(xiàn)類似Express的中間件系統(tǒng)(實(shí)例詳解)
在 Express 中可以給一個(gè)請(qǐng)求設(shè)置若干個(gè)中間件,在處理響應(yīng)時(shí)會(huì)按順序執(zhí)行這些中間件,正在執(zhí)行的中間件可以控制是否執(zhí)行下一個(gè)中間件,這篇文章主要介紹了JavaScript 實(shí)現(xiàn)類似Express的中間件系統(tǒng)的相關(guān)資料,需要的朋友可以參考下2023-02-02