jQuery通過deferred對象管理ajax異步
今天跟大家分享一個jquery中的對象-deferred。其實從jQuery 1.5.0版本開始引入的一個新功能----deferred對象。不過可能在實際開發(fā)過程中用到的并不多,所以沒有太在意。
什么是deferred對象?
開發(fā)網(wǎng)站的過程中,我們經(jīng)常遇到某些耗時很長的javascript操作。其中,既有異步的操作(比如ajax讀取服務(wù)器數(shù)據(jù)),也有同步的操作(比如遍歷一個大型數(shù)組),它們都不是立即能得到結(jié)果的。
通常的做法是,為它們指定回調(diào)函數(shù)(callback)。即事先規(guī)定,一旦它們運行結(jié)束,應該調(diào)用哪些函數(shù)。
但是,在回調(diào)函數(shù)方面,jQuery的功能非常弱。為了改變這一點,jQuery開發(fā)團隊就設(shè)計了deferred對象。
簡單說,deferred對象就是jQuery的回調(diào)函數(shù)解決方案。在英語中,defer的意思是"延遲",所以deferred對象的含義就是"延遲"到未來某個點再執(zhí)行。
這里先不說deferred的概念,我們先看一個例子。
還記得初學的時候,遇到一個實例,先是要ajax請求一個接口(a.json),從返回的數(shù)據(jù)中獲得一個id1值。然后再請求一個接口(b.json)獲得id2,最后需要對這兩個id值同時進行操作。
錯誤解法
那個時候初學,首先想到的方案(現(xiàn)在想想,很傻很天真...)
var id1, id2; $.ajax({ url: 'a.js', dataType: 'json', type: 'get', success: function(d){ id1 = d.item.id; } }); $.ajax({ url: 'b.js', dataType: 'json', type: 'get', success: function(d){ id2 = d.item.id; } }) alert('id1='+id1+','+ 'id2='+ id2);
因為那個時候,還沒有理解異步的概念,所以以為,第二次ajax的時候id已經(jīng)有值了,但是運行之后才發(fā)現(xiàn),變量id其實根本沒被賦值。想要測試上面代碼,點這里
也就是這一刻,我真正明白了:ajax是異步的!?。?。
傻瓜式解法
發(fā)現(xiàn)上面那個方法不能用之后,分析了一下,彈出undefined是因為彈出之前id還沒有被賦值,那我保證在彈出之前給id賦值不就解決了嗎?好的,于是我想到了下面這個方法:
var id1; $.ajax({ url: '/test/json/a.js', dataType: 'json', type: 'get', success: function(d){ id1 = d.item.id; $.ajax({ url: '/test/json/b.js', dataType: 'json', type: 'get', success: function(f){ id2 = f.item.id; alert('id1='+id1+','+ 'id2='+ id2); } }); } })
邏輯雖然正確了,但總覺得怪怪的,如果這里需要嵌套3層呢?4層呢?。。。ajax里面嵌套ajax,如果數(shù)據(jù)很多,訪問速度慢,嵌套更多層,會導致性能下降、影響用戶體驗、代碼不好維護等等問題。所以一般不推薦這種方法??傊@種寫法讓我難以接受。
所以思來想去,覺得不妥。。。然后那個時候就在一個前端群里,詢問各種大牛,直到一個大牛告訴我讓我百度一下deferred,后來認真學習了下,覺得不錯。
使用deferred對象
deferred對象簡介
deferred是jquery中的擴展的一個對象(1.5.0以上的版本支持deferred)。defer的意思是"延遲",所以deferred對象的含義就是"延遲"到未來某個點再執(zhí)行。
簡單說,deferred對象就是jQuery的回調(diào)函數(shù)解決方案。
再簡單說,deferred對象用來管理異步操作,而ajax就是一種異步操作。
deferred基本語法
deferred讓ajax支持新的寫法,代碼如下:
$.ajax({ url: '/test/json/a.js', dataType: 'json', type: 'get' }) .done(function() { alert("成功啦!"); }) .fail(function() { alert("失敗了..."); })
這個大家應該都知道?,F(xiàn)在在編輯器敲入ajax,然后回車,提示的ajax語法結(jié)構(gòu)就是這樣鏈式的寫法。
done函數(shù)就是ajax請求成功的回到函數(shù);
fail函數(shù)就是ajax請求失敗的回調(diào)函數(shù)。
使用deferred的解決方法
var ajax1 = $.ajax({ url: '/test/json/a.js', dataType: 'json', type: 'get' }); var ajax2 = $.ajax({ url: '/test/json/b.js', dataType: 'json', type: 'get', }); $.when(ajax1,ajax2).done(function(d1,d2){ var id1 = d1[0].item.id; var id2 = d2[0].item.id; alert('id1='+id1+', '+ 'id2='+ id2); }).fail(function(){ alert('error'); });
值得一提的是,上面代碼中done函數(shù)的參數(shù),對應的是前面每一個ajax請求返回的數(shù)據(jù)
上面的代碼中,用到了deferred對象的when方法。
它的描述是:
提供一種方法來執(zhí)行一個或多個對象的回調(diào)函數(shù)。
這里的ajax1和ajax2就是deferred對象,done和fail就是回調(diào)函數(shù)。上面代碼的意思是:
只有當兩個ajax請求都成功返回數(shù)據(jù)時,執(zhí)行done函數(shù);只要有一個請求不成功,就執(zhí)行fail函數(shù)。
另外值得一提的是:$.when方法的參數(shù),只支持deferred對象,而ajax返回的就是deferred對象。`
這就已經(jīng)實現(xiàn)了上面的需求了。請求兩個接口,獲得兩個數(shù)據(jù),都成功時,對這兩個數(shù)據(jù)同時進行處理。而且這種鏈式寫法,讓讀者一目了然,而且便于維護擴展。
deferred方法匯總
提到的方法
$.Deferred():生成一個deferred對象。
$.when() 為多個操作指定回調(diào)函數(shù)。
deferred.done():指定操作成功后的回調(diào)函數(shù)
deferred.fail():指定操作失敗后的回調(diào)函數(shù)
未提到的方法
•deferred.resolve()方法和deferred.reject()方法
deferred對象執(zhí)行回調(diào)函數(shù)之前會有一個執(zhí)行狀態(tài)的存在,執(zhí)行狀態(tài)一共有三種———未完成、已完成和已失敗。
未完成狀態(tài),則會繼續(xù)等待,或者執(zhí)行progress()指定的回調(diào)函數(shù)。
已完成狀態(tài),則會執(zhí)行done()方法指定的回調(diào)函數(shù)。
已失敗狀態(tài),則會執(zhí)行fail()方法指定的回調(diào)函數(shù)。
所以這里的deferred.resolve()方法就是手動將deferred對象的狀態(tài)改為已完成,繼而執(zhí)行done方法; deferred.reject()方法就是手動將狀態(tài)改為已失敗,繼而執(zhí)行fail方法。
下面來看一個例子:
var defer = $.Deferred(); // 新建一個Deferred對象 var wait = function(defer){ var tasks = function(){ defer.resolve(); // 改變Deferred對象為已完成狀態(tài) alert("執(zhí)行完畢!"); }; setTimeout(tasks,5000); return defer; }; $.when(wait(defer)) .done(function(){ alert("succeed"); }) .fail(function(){ alert("failed"); });
結(jié)果:等待5秒鐘,先彈出“succeed”,在彈出“執(zhí)行完畢!”。
分析一下代碼執(zhí)行過程:
$.when()里面的參數(shù)是wait函數(shù),也就是一個deferred對象,所以可以繼續(xù)執(zhí)行setTimeout函數(shù),等待5s,執(zhí)行tasks函數(shù),然后手動改變了狀態(tài)為“已完成”,所以執(zhí)行done方法,彈出“succeed”,然后彈出“執(zhí)行完畢!”。
deferred.then():有時為了省事,可以把done()和fail()合在一起寫,這就是then()方法。
function successFun(){ alert("yes"); } function failFun(){ alert('fail'); } $.when($.ajax({ url: '/test/json/a.js', dataType:'json', type: 'get' })) .then(successFun, failFun);
當then方法只有一個參數(shù)時,相當于done方法。當有兩個參數(shù)時,第一個相當于done方法,第二個相當于fail方法。
總結(jié)
deferred對象通過對一個ajax請求的各種回調(diào)函數(shù)的控制,讓jquery寫ajax變的簡單、容易維護、容易擴展。
以上所述是小編給大家介紹的jQuery通過deferred對象管理ajax異步的相關(guān)知識,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
jQuery Ajax 全局調(diào)用封裝實例代碼詳解
這篇文章主要介紹了jQuery Ajax 全局調(diào)用封裝實例代碼詳解的相關(guān)資料,非常不錯具有參考借鑒價值,需要的朋友可以參考下2016-06-06jQuery實現(xiàn)自動與手動切換的滾動新聞特效代碼分享
這篇文章主要介紹了jQuery實現(xiàn)列表自動循環(huán)滾動手動滾動展示新聞,推薦給大家,有需要的小伙伴可以參考下。2015-08-08jQuery Validate 相關(guān)參數(shù)及常用的自定義驗證規(guī)則
這篇文章主要介紹了jQuery Validate 相關(guān)參數(shù)及常用的自定義驗證規(guī)則,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2017-03-03jQuery Selectors(選擇器)的使用(四-五、內(nèi)容篇&可見性篇)
本系列文章主要講述jQuery框架的選擇器(Selectors)使用方法,我將以實例方式進行講述,以簡單,全面為基礎(chǔ),不會涉及很深,我的學習方法:先入門,后進階!2009-12-12