多ajax請(qǐng)求的各類(lèi)解決方案(同步, 隊(duì)列, cancel請(qǐng)求)
•多個(gè)ajax請(qǐng)求相互依賴(lài),必須有先后順序。
•多個(gè)請(qǐng)求被同時(shí)發(fā)送,只需要最后一個(gè)請(qǐng)求。
第1種case
應(yīng)用場(chǎng)景: 這個(gè)場(chǎng)景很多,一個(gè)頁(yè)面打開(kāi)是多個(gè)區(qū)域同時(shí)請(qǐng)求后臺(tái)得到各自的數(shù)據(jù),沒(méi)依賴(lài),沒(méi)順序。
處理方案: 直接用jquery的ajax函數(shù)。這個(gè)用的非常多,這里從略,可看后面的代碼中例子。
第2種case
應(yīng)用場(chǎng)景: 多個(gè)ajax請(qǐng)求,需要順序執(zhí)行,后一個(gè)ajax請(qǐng)求的執(zhí)行參數(shù)是前一個(gè)ajax的結(jié)果。例如: 用戶(hù)登錄后我們發(fā)送一次請(qǐng)求得到用戶(hù)的應(yīng)用ID,然后利用應(yīng)用ID發(fā)送一次請(qǐng)求得到具體的應(yīng)用內(nèi)容(例子雖然不是太恰當(dāng),但基本就是這個(gè)意思了)。
處理方法:
1. 利用ajax參數(shù)async設(shè)置為false,進(jìn)行同步操作。(這個(gè)方法只適合同域操作,跨域需使用下面兩種方法)
2. 利用ajax嵌套(這個(gè)同第1種情況)
3. 利用隊(duì)列進(jìn)行操作
jquery ajax隊(duì)列操作核心代碼:
(function ($) {
var ajaxRequest = {};
$.ajaxQueue = function (settings) {
var options = $.extend({ className: 'DEFEARTNAME' }, $.ajaxSettings, settings);
var _complete = options.complete;
$.extend(options, {
complete: function () {
if (_complete)
_complete.apply(this, arguments);
if ($(document).queue(options.className).length > 0) {
$(document).dequeue(options.className);
} else {
ajaxRequest[options.className] = false;
}
}
});
$(document).queue(options.className, function () {
$.ajax(options);
});
if ($(document).queue(options.className).length == 1 && !ajaxRequest[options.className]) {
ajaxRequest[options.className] = true;
$(document).dequeue(options.className);
}
};
})(jQuery);
第3中case
應(yīng)用場(chǎng)景: 比較典型的是autocomplete控件的操作,這個(gè)我們可以使用第2種情況的處理方法,但我們可能只需要最后次按鍵后返回的結(jié)果,這樣利用第2種處理方法未免有些浪費(fèi)。
處理方法: 保留最后一次請(qǐng)求,cancel之前的請(qǐng)求。
(function ($) {
var jqXhr = {};
$.ajaxSingle = function (settings) {
var options = $.extend({ className: 'DEFEARTNAME' }, $.ajaxSettings, settings);
if (jqXhr[options.className]) {
jqXhr[options.className].abort();
}
jqXhr[options.className] = $.ajax(options);
};
})(jQuery);
對(duì)于這些case都是在多個(gè)ajax請(qǐng)求,響應(yīng)時(shí)間不能控制的情況。下面是完整Demo代碼。
(function ($) {
var jqXhr = {},
ajaxRequest = {};
$.ajaxQueue = function (settings) {
var options = $.extend({ className: 'DEFEARTNAME' }, $.ajaxSettings, settings);
var _complete = options.complete;
$.extend(options, {
complete: function () {
if (_complete)
_complete.apply(this, arguments);
if ($(document).queue(options.className).length > 0) {
$(document).dequeue(options.className);
} else {
ajaxRequest[options.className] = false;
}
}
});
$(document).queue(options.className, function () {
$.ajax(options);
});
if ($(document).queue(options.className).length == 1 && !ajaxRequest[options.className]) {
ajaxRequest[options.className] = true;
$(document).dequeue(options.className);
}
};
$.ajaxSingle = function (settings) {
var options = $.extend({ className: 'DEFEARTNAME' }, $.ajaxSettings, settings);
if (jqXhr[options.className]) {
jqXhr[options.className].abort();
}
jqXhr[options.className] = $.ajax(options);
};
})(jQuery);
var ajaxSleep = (function () {
var _settings = {
type: 'GET',
cache: false,
success: function (msg) {
var thtml = $('#txtContainer').html();
$('#txtContainer').html(thtml + "<br />" + msg);
}
};
return {
get: function (seconds, mode, isAsync) {
var mode = mode || 'ajax',
isAsync = isAsync || false;
$[mode]($.extend(_settings, {
url: "ResponsePage.aspx?second=" + seconds,
async: isAsync,
className: 'GET'
}));
},
post: function (seconds, mode, isAsync) {
var mode = mode || 'ajax',
isAsync = isAsync || false;
$[mode]($.extend(_settings, {
type: 'POST',
url: "PostPage.aspx",
data: { second: seconds },
async: isAsync,
className: 'POST'
}));
}
};
} ());
var launch = function (settings) {
$('#txtContainer').html('');
var mode = settings.mode,
isAsync = settings.isAsync;
ajaxSleep.get(12, mode, isAsync);
ajaxSleep.get(10, mode, isAsync);
ajaxSleep.get(8, mode, isAsync);
ajaxSleep.post(6, mode, isAsync);
ajaxSleep.post(4, mode, isAsync);
ajaxSleep.post(2, mode, isAsync);
}
$(document).ready(function () {
//第1種case
$('#btnLaunchAsync').click(function () {
launch({ isAsync: true });
});
//第2種case
$('#btnLaunchSync').click(function () {
launch({});
});
//第2種case
$('#btnLaunchQueue').click(function () {
launch({ mode: 'ajaxQueue', isAsync: true });
});
//第3種case
$('#btnLaunchSingle').click(function () {
launch({ mode: 'ajaxSingle', isAsync: true });
});
});
default.html
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head id="Head1" runat="server">
<title></title>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.7.1/jquery.min.js" type="text/javascript"></script>
<script type="text/javascript" src="js/default.js"></script>
</head>
<body>
<form id="form1" runat="server">
<input type="button" id="btnLaunchAsync" value="Launch Asynchronous Request" />
<input type="button" id="btnLaunchSync" value="Launch Synchronous Request" />
<input type="button" id="btnLaunchQueue" value="Launch Requested Queue" />
<input type="button" id="btnLaunchSingle" value="Launch Single Request" />
<div id="txtContainer"></div>
</form>
</body>
</html>
PostPage.aspx & ResponsePage.aspx
//ResponsePage.aspx
protected void Page_Load(object sender, EventArgs e)
{
int seconds = int.Parse(Request.QueryString["second"]);
Thread.Sleep(seconds*1000);
Response.Write("GET: selpt for "+ seconds.ToString() +" sec(s)");
}
//PostPage.aspx
protected void Page_Load(object sender, EventArgs e)
{
int seconds = int.Parse(Request.Form["second"]);
Thread.Sleep(seconds * 1000);
Response.Write("POST: selpt for " + seconds.ToString() + " sec(s)");
}
后注: 個(gè)人能力有限,如有錯(cuò)誤敬請(qǐng)指點(diǎn)。這些只是些根據(jù)一些特定情況下的處理,如果一個(gè)ajax請(qǐng)求能解決的問(wèn)題切勿利用兩個(gè)請(qǐng)求來(lái)處理,畢竟需要占用資源。我還是相信沒(méi)有最好的方案,只有最適合的方案。
- 淺析ajax請(qǐng)求json數(shù)據(jù)并用js解析(示例分析)
- Ajax請(qǐng)求中的異步與同步,需要注意的地方說(shuō)明
- AJAX跨域請(qǐng)求json數(shù)據(jù)的實(shí)現(xiàn)方法
- Ajax請(qǐng)求內(nèi)嵌套Ajax請(qǐng)求示例代碼
- 如何將ajax請(qǐng)求返回的Json格式數(shù)據(jù)循環(huán)輸出成table形式
- Ajax請(qǐng)求在數(shù)據(jù)量大的時(shí)候出現(xiàn)超時(shí)的解決方法
- ajax請(qǐng)求亂碼的解決方法(中文亂碼)
- ajax請(qǐng)求成功后新開(kāi)窗口window.open()被攔截解決方法
- ajax 同步請(qǐng)求和異步請(qǐng)求的差異分析
- 在實(shí)戰(zhàn)中可能碰到的幾種ajax請(qǐng)求方法詳解
相關(guān)文章
防止ajax重復(fù)請(qǐng)求的方法(GET和POST)
防止ajax重復(fù)請(qǐng)求的方法(GET和POST) ,需要的朋友可以參考下。2011-10-10jQuery ajax中使用serialize()方法提交表單數(shù)據(jù)示例
使用jQuery ajax的serialize() 方法表單序列化為鍵值對(duì)(key1=value1&key2=value2…)后提交,下面是使用serialize()序列化表單的實(shí)例,感興趣的朋友可以參考下2013-10-10H5移動(dòng)開(kāi)發(fā)Ajax上傳多張Base64格式圖片到服務(wù)器
這篇文章主要為大家詳細(xì)介紹了H5移動(dòng)開(kāi)發(fā)Ajax上傳多張Base64格式圖片到服務(wù)器,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-05-05ajax設(shè)置async校驗(yàn)用戶(hù)名是否存在的實(shí)現(xiàn)方法
下面小編就為大家?guī)?lái)一篇ajax設(shè)置async校驗(yàn)用戶(hù)名是否存在的實(shí)現(xiàn)方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-08-08Ajax 入門(mén)之 GET 與 POST 的不同處詳解
這篇文章主要介紹了Ajax 入門(mén)之 GET 與 POST 的不同處詳解,需要的朋友可以參考下2017-02-02ajax傳遞一個(gè)參數(shù)具體實(shí)現(xiàn)
ajax傳遞一個(gè)參數(shù)或多個(gè)參數(shù)在使用過(guò)程中由于特殊需求經(jīng)常會(huì)用到,下面與大家分享下具體的實(shí)現(xiàn)方法,感興趣的朋友可以參考下哈2013-05-05ajax實(shí)現(xiàn)session不過(guò)期(避免頁(yè)面過(guò)期的現(xiàn)象)
在寫(xiě)博客時(shí)要寫(xiě)好長(zhǎng)時(shí)間但沒(méi)有出現(xiàn)這種情況并且有實(shí)時(shí)的自動(dòng)保存;這就涉及到了session的過(guò)期時(shí)間問(wèn)題,下面與大家分享下具體的實(shí)現(xiàn)方法2013-06-06Ajax案例集下載:新增分頁(yè)查詢(xún)案例(包括《Ajax開(kāi)發(fā)精要》中的兩個(gè)綜合案例) 下載
Ajax案例集下載:新增分頁(yè)查詢(xún)案例(包括《Ajax開(kāi)發(fā)精要》中的兩個(gè)綜合案例) 下載...2007-05-05