中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

jQuery Ajax async=>false異步改為同步時如何解決導致瀏覽器假死的問題

發布時間:2021-07-21 10:57:28 來源:億速云 閱讀:201 作者:小新 欄目:web開發

這篇文章主要介紹jQuery Ajax async=>false異步改為同步時如何解決導致瀏覽器假死的問題,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!

大概代碼如下:

/**
  * 異步當前用戶積分 by zgw 20161216
  * @return {[type]} [description]
 */
 function flushIntegralSum() {
     //點擊按鈕刷新前修改按鈕的文案,已經去掉點擊事情,防止多次點擊
  $("#flushbutton").replaceWith('<a  href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" id="flushbutton">正在刷新</a>');
  $.ajax({
   url:'URL',
   type:'post',
   async:false,
   // data:{},
   success:function(json){
    json = eval('('+json+')');
    if(json.url){window.location.href=json.url;return;}
    $("#flushbutton").replaceWith('<a  href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" onclick="flushFreeSum();" id="flushbutton">刷新積分</a>');
    if(json.code!=1){
     alert(json.msg);
    }else{
     $("#free_sum").html(json.free_sum);
    }
    return;
   }
  });
 }

本以為這么簡單的功能喀喀喀隨便寫寫就沒事了,在運行的時候出現了問題,當用戶點擊刷新積分按鈕時,文案沒有修改為"正在刷新",但是ajax請求發送了,于是我查看網頁代碼,發現js其實把文案和html元素綁定的onclick事件去掉了,在請求成功后有變回原來的了,但是頁面上邊文案沒有改變,當時很奇怪,不知道為什么html代碼里邊改變了,頁面卻沒有變點變化

二、了解問題原因

問題的根源:當時我進行了排查,最后發現是 "async:false" 的問題,換成異步的就沒有問題了,那為什么同步請求會產生代碼失效的問題呢?

原因:瀏覽器的渲染(UI)線程和js線程是互斥的,在執行js耗時操作時,頁面渲染會被阻塞掉。當我們執行異步ajax的時候沒有問題,但當設置為同步請求時,其他的動作(ajax函數后面的代碼,還有渲染線程)都會停止下來。即使我的DOM操作語句是在發起請求的前一句,這個同步請求也會“迅速”將UI線程阻塞,不給它執行的時間。這就是代碼失效的原因。

三、解決問題

1.我當時使用了 setTimeout 來解決,把ajax代碼放在sestTimeout中,讓瀏覽器重啟一個線程來操作,這樣就解決問題了,代碼如下:

function flushIntegralSum() {
     //點擊按鈕刷新前修改按鈕的文案,已經去掉點擊事情,防止多次點擊
  $("#flushbutton").replaceWith('<a  href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" id="flushbutton">正在刷新</a>');
  setTimeout(function(){
   $.ajax({
    url:'URL',
    type:'post',
    async:false,
    // data:{},
    success:function(json){
     json = eval('('+json+')');
     if(json.url){window.location.href=json.url;return;}
     $("#flushbutton").replaceWith('<a  href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" onclick="flushFreeSum();" id="flushbutton">刷新積分</a>');
     if(json.code!=1){
      alert(json.msg);
     }else{
      $("#free_sum").html(json.free_sum);
     }
     return;
    }
   });
  },0) 
 }

setTimeout的第二個參數設為0,瀏覽器會在一個已設的最小時間后執行

到這里問題就解決了,但是你可以試試當你點擊按鈕的時候如果需要彈出一個gif圖片,并且圖片一直在旋轉,提示更新中,你會發現圖片雖然會顯示,但是圖片卻是不動的,那是因為雖然同步請求延遲執行了,但是它執行期間還是會把UI線程給阻塞。這個阻塞相當牛逼,連gif圖片都不動了,看起來像一張靜態圖片一樣。結論很明顯,setTimeout治標不治本,相當于把同步請求“稍稍”異步了一下,接下來還是會進入同步的噩夢,阻塞線程,這種方法只適合發請求之前操作簡單的時間短的情況

2.使用 Deferred 來解決

jQuery在1.5版本之后,引入了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"  src="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>

以上是“jQuery Ajax async=>false異步改為同步時如何解決導致瀏覽器假死的問題”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

西乌珠穆沁旗| 平塘县| 大名县| 宁强县| 台北县| 南召县| 万全县| 随州市| 大英县| 九寨沟县| 苏尼特左旗| 建宁县| 南召县| 罗山县| 濉溪县| 林甸县| 阳朔县| 依兰县| 太保市| 枞阳县| 红河县| 唐山市| 巨野县| 金秀| 泗水县| 贺兰县| 手游| 阿鲁科尔沁旗| 普定县| 马关县| 荣昌县| 三穗县| 梅州市| 开阳县| 揭东县| 永州市| 临颍县| 黄山市| 黎川县| 墨江| 屯昌县|