国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

解決js ajax同步請求造成瀏覽器假死的問題

2019-11-19 14:30:31
字體:
來源:轉載
供稿:網友

一、問題的起因

今天做一個需求遇到了這么個情況,就是用戶個人中心有個功能,點擊按鈕,可以刷新用戶當前的積分,這個肯定需要使用到ajax的同步請求了,當時喀喀喀三下五除二寫玩了,大概代碼如下:

/**  * 異步當前用戶積分 by zgw 20161216  * @return {[type]} [description] */ function flushIntegralSum() {     //點擊按鈕刷新前修改按鈕的文案,已經去掉點擊事情,防止多次點擊  $("#flushbutton").replaceWith('<a style="color:#3fb0ff;font-size:14px;" 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 style="color:#3fb0ff;font-size:14px;" 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 style="color:#3fb0ff;font-size:14px;" 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 style="color:#3fb0ff;font-size:14px;" 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 來解決

以上這篇解決js ajax同步請求造成瀏覽器假死的問題就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 田林县| 胶州市| 高邑县| 同仁县| 琼结县| 靖江市| 霍城县| 吴堡县| 天祝| 灌阳县| 思茅市| 康保县| 洛宁县| 鹤峰县| 舞钢市| 绥中县| 偏关县| 浪卡子县| 阿克陶县| 交口县| 韶关市| 彰化县| 徐汇区| 九龙县| 砀山县| 府谷县| 吴忠市| 锡林浩特市| 宣化县| 敖汉旗| 卓资县| 西宁市| 屏东县| 汉阴县| 内乡县| 扶余县| 丹江口市| 绥德县| 湾仔区| 河间市| 闻喜县|