Jquery ajax 同步阻塞引起的UI線程阻塞問題
最近做一個項目,遇到了一個問題同步ajax引起的ui線程阻塞問題,下面把我的問題解決過程分享給大家。
事情起因是這樣的,因為頁面上有多個相似的異步請求動作,本著提高代碼可重用性的原則,我封裝了一個名為getData的函數,它接收不同參數,只負責獲取數據,然后把數據return?;镜倪壿媱冸x出來是這樣的:
function getData1(){ var result; $.ajax({ url : "p.php", async : false, success: function(data){ result = data; } }); return result; }
這里的ajax不能用異步的,否則函數返回時,result還未賦值,會出錯。所以我加了async:false??雌饋砗孟駴]什么問題。我調用這個函數可以正常的得到數據。
$(".btn1").click(function(){ var data = getData1(); alert(data); });
接下來,要加另外一個功能,由于ajax請求有一定的耗時,所以我需要在發(fā)出請求前頁面有個loading效果,即顯示一張“正在加載”的gif圖片,想必大家也都見過。所以我的處理函數就變成了這樣:
$(".btn1").click(function(){ $(".loadingicon").show(); var data = getData1(); $(".loadingicon").hide(); alert(data); });
請求之前顯示loading圖片,請求完成后把它隱藏??雌饋硪矝]什么問題。為了看清效果,我的p.php代碼sleep了3秒,如下:
<?php sleep(3); echo ("aaaaaa"); ?>
但是我運行的時候問題出現了,我點擊按鈕并未像預想的那樣出現這個loading圖片,頁面什么反應也沒有。排除良久找到了原因,就在async:false這里。
瀏覽器的渲染(UI)線程和js線程是互斥的,在執(zhí)行js耗時操作時,頁面渲染會被阻塞掉。當我們執(zhí)行異步ajax的時候沒有問題,但當設置為同步請求時,其他的動作(ajax函數后面的代碼,還有渲染線程)都會停止下來。即使我的DOM操作語句是在發(fā)起請求的前一句,這個同步請求也會“迅速”將UI線程阻塞,不給它執(zhí)行的時間。這就是代碼失效的原因。
setTimeout解決阻塞問題
既然明白了問題在哪里,我們就來針對性想辦法。為了不讓同步ajax請求阻塞線程,我想到了setTimeout,把請求的代碼放到sestTimeout中,讓瀏覽器重啟一個線程來操作,不就解決問題了嗎?于是乎,我的代碼就變成了這樣:
$(".btn2").click(function(){ $(".loadingicon").show(); setTimeout(function(){ $.ajax({ url : "p.php", async : false, success: function(data){ $(".loadingicon").hide(); alert(data); } }); }, 0); });
setTimeout的第二個參數設為0,瀏覽器會在一個已設的最小時間后執(zhí)行。不管三七二十一先運行起來看看。
結果loading圖片顯示出來了,但是?。?!圖片怎么不動呢,我明明是一張動態(tài)gif圖。這個時候我很快就想到了,雖然同步請求延遲執(zhí)行了,但是它執(zhí)行期間還是會把UI線程給阻塞。這個阻塞相當牛逼,連gif圖片都不動了,看起來像一張靜態(tài)圖片一樣。
結論很明顯,setTimeout治標不治本,相當于把同步請求“稍稍”異步了一下,接下來還是會進入同步的噩夢,阻塞線程。方案失敗。
是時候用Deferred了
jQuery在1.5版本之后,引入了Deferred對象,提供的很方便的廣義異步機制。詳情可參看阮一峰老師的這篇文章http://www.ruanyifeng.com/blog/2011/08/a_detailed_explanation_of_jquery_deferred_object.html。
于是我用Deferred對象改寫了代碼,如下:
function getData3(){ var defer = $.Deferred(); $.ajax({ url : "p.php", //async : false, success: function(data){ defer.resolve(data) } }); return defer.promise(); } $(".btn3").click(function(){ $(".loadingicon").show(); $.when(getData3()).done(function(data){ $(".loadingicon").hide(); alert(data); }); });
可以看到我在ajax請求中去掉了async:false,也就是說,這個請求又是異步的了。另外請注意success函數中的這一 句:defer.resolve(data),Deferred對象的resolve方法可傳入一個參數,任意類型。這個參數可以在done方法中拿到, 所以我們異步請求來的數據就可以以這樣的方式來返回了。
至此,問題得到了解決。Deferred對象如此強大且方便,我們可以好好利用它。
我的全部測試代碼如下,有意的同學可以拿去測一下:
<button class="btn1">async:false</button> <button class="btn2">setTimeout</button> <button class="btn3">deferred</button> <img class="loadingicon" style="position:fixed;left:50%;top:50%;margin-left:-16px;margin-top:-16px;display:none;" src=http://www.update8.com/Web/Jquery/"loading2.gif" alt="正在加載" /> <script> function getData1(){ var result; $.ajax({ url : "p.php", async : false, success: function(data){ result = data; } }); return result; } $(".btn1").click(function(){ $(".loadingicon").show(); var data = getData1(); $(".loadingicon").hide(); alert(data); }); $(".btn2").click(function(){ $(".loadingicon").show(); setTimeout(function(){ $.ajax({ url : "p.php", async : false, success: function(data){ $(".loadingicon").hide(); alert(data); } }); }, 0); }); function getData3(){ var defer = $.Deferred(); $.ajax({ url : "p.php", //async : false, success: function(data){ defer.resolve(data) } }); return defer.promise(); } $(".btn3").click(function(){ $(".loadingicon").show(); $.when(getData3()).done(function(data){ $(".loadingicon").hide(); alert(data); }); });</script>
ps:$.ajax的參數描述
參數 描述
url 必需。規(guī)定把請求發(fā)送到哪個 URL。
data 可選。映射或字符串值。規(guī)定連同請求發(fā)送到服務器的數據。
success(data, textStatus, jqXHR) 可選。請求成功時執(zhí)行的回調函數。
dataType
可選。規(guī)定預期的服務器響應的數據類型。
默認執(zhí)行智能判斷(xml、json、script 或 html)。
相關文章
jQuery插件jcrop+Fileapi完美實現圖片上傳+裁剪+預覽的代碼分享
這篇文章主要介紹了jQuery插件jcrop+Fileapi完美實現圖片上傳+裁剪+預覽的代碼,非常的簡單實用,效果也很棒,有需要的小伙伴可以參考下。2015-04-04jQuery實現右側顯示可向左滑動展示的深色QQ客服效果代碼
這篇文章主要介紹了jQuery實現右側顯示可向左滑動展示的深色QQ客服效果代碼,涉及jQuery控制頁面元素樣式動態(tài)變換的實現技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-10-10ie8模式下click無反應點擊option無反應的解決方法
點擊select里面的option,將其賦值到上面的input,直接用jQuery寫的,問題是在用IE8打開的時候,點擊option沒有任何反應2014-10-10Jquery+WebService 校驗賬號是否已被注冊的代碼
在Javascirpt代碼中,調用Jquery的方法$.Ajax(function)實現Ajax,傳遞賬號信息給Web服務,Web服務再調用數據庫操作類查詢數據庫,并返回數據給前臺頁面。2010-07-07