jQuery的promise與deferred對(duì)象在異步回調(diào)中的作用
來源:易賢網(wǎng) 閱讀:581 次 日期:2016-07-06 10:42:57
溫馨提示:易賢網(wǎng)小編為您整理了“jQuery的promise與deferred對(duì)象在異步回調(diào)中的作用”,方便廣大網(wǎng)友查閱!

這篇文章主要介紹了jQuery的promise與deferred對(duì)象在異步回調(diào)中的作用,需要的朋友可以參考下

一、前言

為了讓前端們從回調(diào)的地獄中回到天堂, jQuery 也引入了 Promise 的概念。 Promise 是一種令代碼異步行為更加優(yōu)雅的抽象,有了它,我們就可以像寫同步代碼一樣去寫異步代碼。 jQuery 從1.5版本開始實(shí)現(xiàn)了 CommonJS Promise/A 規(guī)范這一重量級(jí)方案,不過沒有嚴(yán)格按照規(guī)范進(jìn)行實(shí)現(xiàn),有一些API上的差異。

好,讓我們來看看他們的特性吧( 本文示例基于jquery 1.8版本以上 )。

二、示例

以前寫動(dòng)畫時(shí),我們通常是這么干的:

$('.animateEle').animate({

 opacity:'.5'

}, 4000,function(){

 $('.animateEle2').animate({

  width:'100px'

 },2000,function(){

  // 這樣太傷了

  $('.animateEle3').animate({

   height:'0'

  },2000);

 });

});

假如這么使用回調(diào)的話,那就太傷了。幸好,還有一些現(xiàn)成的 Promise 解決方案來優(yōu)雅地解決這種問題。

我們看看 jQuery 提供的解決辦法。

var animate1 = function() {

 return $('.animateEle1').animate({opacity:'.5'},4000).promise();

};

var animate2 = function() {

 return $('.animateEle2').animate({width:'100px'},2000).promise();

};

var animate3 = function(){

 return $('.animateEle3').animate({height:'0'},2000).promise();

};

// so easy,有木有,so clear,有木有

$.when(animate1()).then(animate2).then(animate3);

很明顯,更改后的代碼更易懂易讀了。

但是,上面的代碼,有些細(xì)節(jié)的東西并沒有透露,一不小心,就容易出錯(cuò),得不到我們想要的順序完成動(dòng)畫的效果。下面讓我們來全面理解 jQuery 提供的 promise 和 deferred 對(duì)象的方法,看看到底如何使用。

三、promise和deffered對(duì)象方法

promise 對(duì)象其實(shí)就是 deferred 對(duì)象的特例,因?yàn)?promise 對(duì)象不能更改異步狀態(tài),而 deferred 對(duì)象可以。這點(diǎn)在他們的方法設(shè)計(jì)上,有著明顯的體現(xiàn)。

1.promise對(duì)象方法

通常,對(duì)于DOM,動(dòng)畫,ajax相關(guān)方法,我們都可以使用 promise 方法。調(diào)用 promise 方法,返回的是 promise 對(duì)象??梢枣?zhǔn)秸{(diào)用 promise 方法。

promise對(duì)象常見的方法有三個(gè) : done , fail , then 。

其它的方法就不要去記了, jquery 這里的接口方法太多了,在我看來挺啰嗦的,就跟早期的事件方法綁定一樣, live , delegate , bind ,最終不是都?xì)w為 on 來管了么。

代碼示例,如下:

(1)DOM使用 promise 方法:

var box=$('#box');

box.promise().done(function(ele){

  console.log(ele);//jQuery box

});

(2)Ajax使用 promise 方法(默認(rèn)返回一個(gè) promise 對(duì)象,所以可以不必顯式調(diào)用 promise 方法):

$.post('/',{}).done(function(data){

  console.log('請(qǐng)求成功');

}).fail(function(){

  console.log('請(qǐng)求錯(cuò)誤');

});

動(dòng)畫示例已有,就不重復(fù)列出了。

2.deferred對(duì)象方法

那么Deferred和Promise之間有什么區(qū)別呢?正如你在前面看到的,一個(gè)promise就是一個(gè)由異步函數(shù)返回的對(duì)象。當(dāng)你想要自己編寫一個(gè)這樣的函數(shù)時(shí)你需要使用一個(gè)deferred。

一個(gè)deferred對(duì)象能做的和一個(gè)promise對(duì)象差不多,但是它有兩個(gè)函數(shù)來觸發(fā)done()和fail()函數(shù)。

一個(gè)deferred對(duì)象擁有一個(gè)resolve()函數(shù)來處理一個(gè)成功的結(jié)果并執(zhí)行與done()相關(guān)的函數(shù)。reject()函數(shù)則用來處理失敗的結(jié)果并執(zhí)行與fail()相關(guān)的函數(shù)。

你可以給resolve()和reject()函數(shù)都提供參數(shù),然后它們都將傳遞給與done()和fail()相關(guān)的回調(diào)函數(shù)。

promise對(duì)象沒有resolve()和reject()函數(shù)。這是因?yàn)槟銓romise放到了其他的腳本中并且你也不想promise去resolve或者reject一個(gè)promise。

下面是一個(gè)關(guān)于deferred的簡(jiǎn)單例子。html僅僅是一個(gè)簡(jiǎn)單的擁有id屬性為”result”的空div。

$('#result').html('waiting...');

var promise = wait();

promise.done(result);

function result() {

 $('#result').html('done');

}

function wait() {

 var deferred = $.Deferred();

 setTimeout(function() {

  deferred.resolve();

 }, 2000);

 return deferred.promise();

}  

其中,wait()函數(shù)返回了一個(gè)promise。它將在2s之后被解析。除了setTimeout之外,異步函數(shù)中所有的東西都能這樣使用,比如 動(dòng)畫,Web worker等等。wait()函數(shù)中的代碼應(yīng)該很清晰,我們使用了deferred對(duì)象,但是我們返回了一個(gè)限制的promise對(duì)象。

對(duì)于 deferred 對(duì)象呢,也就是使用 $.Deferred() 方法,以及 $.when() 等方法創(chuàng)造出來的對(duì)象,有如下的常用方法:

resolve , reject , notify ;

done , fail , progress ;

另外還有 promise 、 then 和 always 方法。

之所以這么排版,是因?yàn)樗麄兪菍?duì)應(yīng)的,也就是說: resolve 方法會(huì)觸發(fā) done 的回調(diào)執(zhí)行, reject 會(huì)觸發(fā) fail 的回調(diào), notify 會(huì)觸發(fā) progress 的回調(diào)。

直接看代碼:

var wait = function(ms) {

 var dtd = $.Deferred();

 setTimeout(dtd.resolve, ms);

 // setTimeout(dtd.reject, ms);

 // setTimeout(dtd.notify, ms);

 return dtd.promise(); //此處也可以直接返回dtd

};

wait(2500).done(function() {

 console.log('haha,師太,你可讓老衲久等了');

}).fail(function() {

 console.log('失敗了');

}).progress(function(res) {

 console.log('等待中...');

});

我們看到了,上面的代碼中,在 wait 函數(shù)中,返回的是個(gè) promise 對(duì)象,而不是 deferred 對(duì)象。

要知道, promise 對(duì)象是沒有 resolve , reject , notify 等方法的,也就意味著,你無法針對(duì) promise 對(duì)象進(jìn)行狀態(tài)更改,只能在 done 或 fail 中進(jìn)行回調(diào)配置。所以,你如果這么調(diào)用 wait(2500).resolve() 將會(huì)報(bào)錯(cuò),因?yàn)?wait(2500) 返回的是個(gè) promise 對(duì)象,不存在 resolve 方法。

但是,這么做,有個(gè)好處,我們把 dtd 這個(gè) deferred 對(duì)象放在了 wai t函數(shù)中,作為了局部變量,避免了全局的污染;進(jìn)一步通過 promise 方法,轉(zhuǎn)化 dtd 這個(gè) deferred 對(duì)象為 promise 對(duì)象,避免了函數(shù) wait 外部可能發(fā)生的狀態(tài)更改(假如我們確實(shí)有這個(gè)需求)。

比如:

var wait = function(ms) {

 var dtd = $.Deferred();

 setTimeout(dtd.resolve, ms);

 // setTimeout(dtd.reject, ms);

 // setTimeout(dtd.notify, ms);

 return dtd; //此處也可以直接返回dtd

};

wait(2500).reject().fail(function(){

 console.log('失敗了...............');

});

我們?cè)谕獠扛牧?wait 返回的 deferred 對(duì)象的狀態(tài),這樣必然觸發(fā)該對(duì)象的 fail 回調(diào)函數(shù)。

對(duì)于 always 方法,從字面意思上就很容易理解, deferred 對(duì)象無論是 resolve 還是 reject ,都會(huì)觸發(fā)該方法的回調(diào)。

3.其它共性

此處講講 then 和 $.when 方法的使用。它們對(duì) promise 對(duì)象也適用。

$.when 方法接受多個(gè) deferred 對(duì)象或者純javascript對(duì)象,返回 promise 對(duì)象。

then 方法依次接受三個(gè)回調(diào),分別為 deferred 對(duì)象 resolve , reject , notify 后觸發(fā)的回調(diào),返回一個(gè) promise 對(duì)象。注意,必須傳入函數(shù),而該函數(shù)只有返回一個(gè) promise 對(duì)象,才能夠讓異步事件按照預(yù)期順序來執(zhí)行。

我們來看看最開始的動(dòng)畫示例代碼, $.when(animate1()).then(animate2).then(animate3) , $.when 方法中接受了一個(gè) animate1 的函數(shù)執(zhí)行結(jié)果,也就是得到了一個(gè) promise 對(duì)象,而后的 then 中,則只是接受了一個(gè)變量名,這樣得到的結(jié)果是一個(gè)匿名的函數(shù)體,而該函數(shù)中返回的是 promise 對(duì)象。正好符合了我們對(duì) then 接受參數(shù)的要求。

假如我們把執(zhí)行語句改成: $.when(animate1()).then(animate2()).then(animate3()) ,這樣造成的結(jié)果就是三個(gè)動(dòng)畫同步執(zhí)行了。與 $.when(animate1(),animate2(),animate3()) 無異。

既然 then 是如此要求,那么與 then 方法類似的 done , fail , progress 也是一樣的。

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

2025國考·省考課程試聽報(bào)名

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