讓ajax更加友好的實(shí)現(xiàn)方法(實(shí)時(shí)顯示后臺(tái)處理進(jìn)度。)
更新時(shí)間:2010年06月02日 16:39:24 作者:
我們能否讓ajax更加友好點(diǎn),實(shí)時(shí)顯示服務(wù)器處理的進(jìn)度了?這在一些長(zhǎng)時(shí)間的請(qǐng)求中尤其重要,比如上傳文件、發(fā)送郵件、批量處理數(shù)據(jù)。
ajax應(yīng)用越來(lái)越多,大部分ajax處理都是在前臺(tái)顯示1個(gè)"loading...",然后把數(shù)據(jù)提交給服務(wù)器進(jìn)行處理,處理完畢后顯示"處理完畢"。我們能否讓ajax更加友好點(diǎn),實(shí)時(shí)顯示服務(wù)器處理的進(jìn)度了?這在一些長(zhǎng)時(shí)間的請(qǐng)求中尤其重要,比如上傳文件、發(fā)送郵件、批量處理數(shù)據(jù)。答案當(dāng)然是可以的,不然就不會(huì)寫這個(gè)了,對(duì)吧,^_^。
存在的問題:
要解決實(shí)現(xiàn)上面的功能,需要解決下面幾個(gè)問題:
1. 服務(wù)器如何在處理一部分?jǐn)?shù)據(jù)后傳遞部分response到瀏覽器。
2、瀏覽器如何能處理服務(wù)器傳遞過(guò)來(lái)部分?jǐn)?shù)據(jù),并保持http連接直到處理完全完畢。
要解決第1個(gè)問題,使用flush讓response分塊進(jìn)行呈現(xiàn)就可以了,具體請(qǐng)參考我另一遍隨筆"flush讓頁(yè)面分塊,逐步呈現(xiàn)";
第2個(gè)問題,則需要用到XMLHttpRequest的readyState狀態(tài),w3c對(duì) readyState 定義如下幾個(gè)值:
UNSENT = 0; // 沒有發(fā)送請(qǐng)求
OPENED = 1; // 已經(jīng)打開http連接
HEADERS_RECEIVED = 2; // 接收到response header
LOADING = 3; // 真正接收response body
DONE = 4; // 請(qǐng)求接收完畢
相信狀態(tài)4大家是天天在用,而我們這里需要用到就是狀態(tài)3。
實(shí)例:
廢話少說(shuō),代碼實(shí)例比什么文字解釋都管用。我們這里假設(shè)服務(wù)器的1個(gè)處理需要6秒種,每秒種處理1條記錄,總共處理6條記錄,我們需要服務(wù)器每處理完1條數(shù)據(jù),客戶端則顯示處理進(jìn)度(包括文字和進(jìn)度條)。
服務(wù)器端代碼(下面JSP代碼):
<%
// 下面設(shè)置Content-Type:application/x-javascript 是為了適應(yīng)Webkit的瀏覽器(chrome,safari)
response.setHeader("Content-Type","application/x-javascript");
int count = 6; // 處理6條數(shù)據(jù)
for(int i=0;i<count;i++){
// 處理完畢一條,輸出結(jié)果到客戶端
out.println(i+1);
out.flush();
// 這里假設(shè)每條數(shù)據(jù)處理時(shí)間為1秒
Thread.currentThread().sleep(1000);
}
%>
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>
<style>
#divProgress{width:300px;height:24px;position:relative;}
#divProgress div{position:absolute;left:0;top:0;height:24px;}
#progressBg{background-color:#B9F8F9;z-index:10;}
#progressText{z-index:15;text-align:center;width:100%;}
</style>
</head>
<body>
<div id="divProgress">
<div id="progressBg"></div>
<div id="progressText"></div>
</div>
<br />
<button onclick="send()">提交數(shù)據(jù)</button>
<script>
var t = document.getElementById("progressText");
var bg = document.getElementById("progressBg");
function send(){
t.innerHTML = "loading...";
bg.style.width = "0px";
var xhr = new window.XMLHttpRequest();
if(!window.XMLHttpRequest){
try {
xhr = new window.ActiveXObject("Microsoft.XMLHTTP");
} catch(e) {}
}
xhr.open("post","http://localhost:801/ChunkTest/chunk.jsp?count=6");
var oldSize=0;
xhr.onreadystatechange = function(){
if(xhr.readyState > 2){
var tmpText = xhr.responseText.substring(oldSize);
oldSize = xhr.responseText.length;
if(tmpText.length > 0 ){
// 設(shè)置文本
t.innerHTML = tmpText + "/6";
// 設(shè)置進(jìn)度條
var width = parseInt(tmpText)/6*300;
bg.style.width = width+"px";
}
}
if(xhr.readyState == 4){
// 請(qǐng)求執(zhí)行完畢
t.innerHTML = "執(zhí)行完畢";
bg.style.width = "300px";
}
}
xhr.send(null);
}
</script>
</body>
</html>
運(yùn)行效果圖:


缺點(diǎn):
看到這里或許你已經(jīng)蠢蠢欲動(dòng),想自己動(dòng)手試試了。但是注意上面的方法雖好,但也有個(gè)缺點(diǎn),就是瀏覽器的支持問題。目前IE所有版本的瀏覽器都不支持 xhr.readyState == 3狀態(tài),IE瀏覽器不支持在response響應(yīng)完畢前讀取responseText屬性。 具體可查看MSDN : XMLHttpRequest Object
基于Webkit的瀏覽器支持的不是很好,需要設(shè)置Content-Type:application/x-javascript才行(經(jīng)測(cè)試發(fā)現(xiàn)Content-Type:text/html在有些情況下正常,有些情況下又不正常,而用application/x-javascript都正常)。
看到了缺點(diǎn)后是否又打擊了你的積極性了,其實(shí)針對(duì)IE,我們不需要做太多處理,IE不支持,就不會(huì)顯示進(jìn)度,就變成跟傳統(tǒng)的ajax請(qǐng)求一樣,一直顯示1個(gè)loading直到請(qǐng)求完畢。我們只需要加1個(gè)簡(jiǎn)單的判斷,判斷如果是ie則不執(zhí)行xhr.readyState > 2中的代碼,如果不加判斷,IE下會(huì)報(bào)JS錯(cuò)誤.
DEMO:
demo服務(wù)器不太好,而且在國(guó)外,隨時(shí)可能會(huì)點(diǎn)擊不了,而且有時(shí)候運(yùn)行效果不是很好,大家知曉下,最好是把代碼copy到本地進(jìn)行測(cè)試.
請(qǐng)使用firefox或chrome查看demo,ie查看的效果跟一般的ajax沒什么不一樣.
http://213.186.44.204:8080/ChunkTest/index.html
轉(zhuǎn)載請(qǐng)注明出處:http://www.cnblogs.com/BearsTaR/。 禁止商用!
存在的問題:
要解決實(shí)現(xiàn)上面的功能,需要解決下面幾個(gè)問題:
1. 服務(wù)器如何在處理一部分?jǐn)?shù)據(jù)后傳遞部分response到瀏覽器。
2、瀏覽器如何能處理服務(wù)器傳遞過(guò)來(lái)部分?jǐn)?shù)據(jù),并保持http連接直到處理完全完畢。
要解決第1個(gè)問題,使用flush讓response分塊進(jìn)行呈現(xiàn)就可以了,具體請(qǐng)參考我另一遍隨筆"flush讓頁(yè)面分塊,逐步呈現(xiàn)";
第2個(gè)問題,則需要用到XMLHttpRequest的readyState狀態(tài),w3c對(duì) readyState 定義如下幾個(gè)值:
UNSENT = 0; // 沒有發(fā)送請(qǐng)求
OPENED = 1; // 已經(jīng)打開http連接
HEADERS_RECEIVED = 2; // 接收到response header
LOADING = 3; // 真正接收response body
DONE = 4; // 請(qǐng)求接收完畢
相信狀態(tài)4大家是天天在用,而我們這里需要用到就是狀態(tài)3。
實(shí)例:
廢話少說(shuō),代碼實(shí)例比什么文字解釋都管用。我們這里假設(shè)服務(wù)器的1個(gè)處理需要6秒種,每秒種處理1條記錄,總共處理6條記錄,我們需要服務(wù)器每處理完1條數(shù)據(jù),客戶端則顯示處理進(jìn)度(包括文字和進(jìn)度條)。
服務(wù)器端代碼(下面JSP代碼):
復(fù)制代碼 代碼如下:
<%
// 下面設(shè)置Content-Type:application/x-javascript 是為了適應(yīng)Webkit的瀏覽器(chrome,safari)
response.setHeader("Content-Type","application/x-javascript");
int count = 6; // 處理6條數(shù)據(jù)
for(int i=0;i<count;i++){
// 處理完畢一條,輸出結(jié)果到客戶端
out.println(i+1);
out.flush();
// 這里假設(shè)每條數(shù)據(jù)處理時(shí)間為1秒
Thread.currentThread().sleep(1000);
}
%>
html代碼:
復(fù)制代碼 代碼如下:
<!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>
<style>
#divProgress{width:300px;height:24px;position:relative;}
#divProgress div{position:absolute;left:0;top:0;height:24px;}
#progressBg{background-color:#B9F8F9;z-index:10;}
#progressText{z-index:15;text-align:center;width:100%;}
</style>
</head>
<body>
<div id="divProgress">
<div id="progressBg"></div>
<div id="progressText"></div>
</div>
<br />
<button onclick="send()">提交數(shù)據(jù)</button>
<script>
var t = document.getElementById("progressText");
var bg = document.getElementById("progressBg");
function send(){
t.innerHTML = "loading...";
bg.style.width = "0px";
var xhr = new window.XMLHttpRequest();
if(!window.XMLHttpRequest){
try {
xhr = new window.ActiveXObject("Microsoft.XMLHTTP");
} catch(e) {}
}
xhr.open("post","http://localhost:801/ChunkTest/chunk.jsp?count=6");
var oldSize=0;
xhr.onreadystatechange = function(){
if(xhr.readyState > 2){
var tmpText = xhr.responseText.substring(oldSize);
oldSize = xhr.responseText.length;
if(tmpText.length > 0 ){
// 設(shè)置文本
t.innerHTML = tmpText + "/6";
// 設(shè)置進(jìn)度條
var width = parseInt(tmpText)/6*300;
bg.style.width = width+"px";
}
}
if(xhr.readyState == 4){
// 請(qǐng)求執(zhí)行完畢
t.innerHTML = "執(zhí)行完畢";
bg.style.width = "300px";
}
}
xhr.send(null);
}
</script>
</body>
</html>
運(yùn)行效果圖:


缺點(diǎn):
看到這里或許你已經(jīng)蠢蠢欲動(dòng),想自己動(dòng)手試試了。但是注意上面的方法雖好,但也有個(gè)缺點(diǎn),就是瀏覽器的支持問題。目前IE所有版本的瀏覽器都不支持 xhr.readyState == 3狀態(tài),IE瀏覽器不支持在response響應(yīng)完畢前讀取responseText屬性。 具體可查看MSDN : XMLHttpRequest Object
基于Webkit的瀏覽器支持的不是很好,需要設(shè)置Content-Type:application/x-javascript才行(經(jīng)測(cè)試發(fā)現(xiàn)Content-Type:text/html在有些情況下正常,有些情況下又不正常,而用application/x-javascript都正常)。
看到了缺點(diǎn)后是否又打擊了你的積極性了,其實(shí)針對(duì)IE,我們不需要做太多處理,IE不支持,就不會(huì)顯示進(jìn)度,就變成跟傳統(tǒng)的ajax請(qǐng)求一樣,一直顯示1個(gè)loading直到請(qǐng)求完畢。我們只需要加1個(gè)簡(jiǎn)單的判斷,判斷如果是ie則不執(zhí)行xhr.readyState > 2中的代碼,如果不加判斷,IE下會(huì)報(bào)JS錯(cuò)誤.
DEMO:
demo服務(wù)器不太好,而且在國(guó)外,隨時(shí)可能會(huì)點(diǎn)擊不了,而且有時(shí)候運(yùn)行效果不是很好,大家知曉下,最好是把代碼copy到本地進(jìn)行測(cè)試.
請(qǐng)使用firefox或chrome查看demo,ie查看的效果跟一般的ajax沒什么不一樣.
http://213.186.44.204:8080/ChunkTest/index.html
轉(zhuǎn)載請(qǐng)注明出處:http://www.cnblogs.com/BearsTaR/。 禁止商用!
您可能感興趣的文章:
- AJAX和JSP實(shí)現(xiàn)的基于WEB的文件上傳的進(jìn)度控制代碼
- AJAX 進(jìn)度條實(shí)現(xiàn)代碼
- ajax+php打造進(jìn)度條 readyState各狀態(tài)
- ajax+php打造進(jìn)度條代碼[readyState各狀態(tài)說(shuō)明]
- ajax提交加載進(jìn)度條示例代碼
- 一個(gè)簡(jiǎn)單的ajax上傳進(jìn)度顯示示例
- php+ajax實(shí)現(xiàn)帶進(jìn)度條的大數(shù)據(jù)排隊(duì)導(dǎo)出思路以及源碼
- jquery+php+ajax顯示上傳進(jìn)度的多圖片上傳并生成縮略圖代碼
- jsp+ajax發(fā)送GET請(qǐng)求的方法
- jquery+ajax實(shí)現(xiàn)跨域請(qǐng)求的方法
- asp.net+ajax的Post請(qǐng)求實(shí)例
- 服務(wù)端配置實(shí)現(xiàn)AJAX跨域請(qǐng)求
- 解決ajax跨域請(qǐng)求數(shù)據(jù)cookie丟失問題
- django通過(guò)ajax發(fā)起請(qǐng)求返回JSON格式數(shù)據(jù)的方法
- Ajax在請(qǐng)求過(guò)程中顯示進(jìn)度的簡(jiǎn)單實(shí)現(xiàn)
相關(guān)文章
添加后臺(tái)list給前臺(tái)select標(biāo)簽賦值簡(jiǎn)單實(shí)現(xiàn)
本文為大家介紹下通過(guò)添加后臺(tái)list怎么實(shí)現(xiàn)給前臺(tái)select標(biāo)簽賦值,具體代碼如下,感興趣的朋友可以參考下哈2013-07-07ajax從JSP傳遞對(duì)象數(shù)組到后臺(tái)的方法
今天小編就為大家分享一篇ajax從JSP傳遞對(duì)象數(shù)組到后臺(tái)的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-08-08使用Ajax時(shí)處理用戶session失效問題的解決方法
這篇文章主要為大家詳細(xì)介紹了使用Ajax時(shí)處理用戶session失效問題的解決方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-03-03json獲取數(shù)據(jù)庫(kù)的信息在前端頁(yè)面顯示方法
今天小編就為大家分享一篇json獲取數(shù)據(jù)庫(kù)的信息在前端頁(yè)面顯示方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-05-05Ajax實(shí)現(xiàn)三級(jí)聯(lián)動(dòng)效果
這篇文章主要為大家詳細(xì)介紹了Ajax實(shí)現(xiàn)三級(jí)聯(lián)動(dòng)效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-10-10Ajax實(shí)現(xiàn)動(dòng)態(tài)顯示并操作表信息的方法
今天小編就為大家分享一篇Ajax實(shí)現(xiàn)動(dòng)態(tài)顯示并操作表信息的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-08-08利用Ajax實(shí)現(xiàn)在腳本里傳值實(shí)例介紹
Ajax實(shí)現(xiàn)在腳本里傳值可以解決實(shí)際上的一些問題,本文實(shí)現(xiàn)了一下,感興趣的朋友可以參考下,希望可以幫助到你2013-04-04