解决js ajax同步请求造成浏览器假死的问题


Posted in Javascript onJanuary 18, 2018

一、问题的起因

今天做一个需求遇到了这么个情况,就是用户个人中心有个功能,点击按钮,可以刷新用户当前的积分,这个肯定需要使用到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同步请求造成浏览器假死的问题就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
Add Formatted Text to a Word Document
Jun 15 Javascript
Extjs中使用extend(js继承) 的代码
Mar 15 Javascript
AngularJS基础 ng-cloak 指令简单示例
Aug 01 Javascript
利用js获取下拉框中所选的值
Dec 01 Javascript
jQuery模拟窗口抖动效果
Mar 15 Javascript
微信小程序五星评分效果实现代码
Apr 06 Javascript
vue.js之vue-cli脚手架的搭建详解
May 05 Javascript
ES7之Async/await的使用详解
Mar 28 Javascript
vue 微信扫码登录(自定义样式)
Jan 06 Javascript
Vue 图片压缩并上传至服务器功能
Jan 15 Javascript
从零开始用webpack构建一个vue3.0项目工程的实现
Sep 24 Javascript
vue使用lodop打印控件实现浏览器兼容打印的方法
Feb 07 Vue.js
js实时监控文本框输入字数的实例代码
Jan 18 #Javascript
JavaScript实现删除数组重复元素的5种常用高效算法总结
Jan 18 #Javascript
react-router4 配合webpack require.ensure 实现异步加载的示例
Jan 18 #Javascript
分享ES6的7个实用技巧
Jan 18 #Javascript
vue 动态修改a标签的样式的方法
Jan 18 #Javascript
详解vue-meta如何让你更优雅的管理头部标签
Jan 18 #Javascript
Nuxt.js踩坑总结分享
Jan 18 #Javascript
You might like
php download.php实现代码 跳转到下载文件(response.redirect)
2009/08/26 PHP
色色整理的PHP面试题集锦
2012/03/08 PHP
PHP关联链接常用代码
2012/11/05 PHP
学习php设计模式 php实现装饰器模式(decorator)
2015/12/07 PHP
javascript 通用简单的table选项卡实现
2010/05/07 Javascript
Javascript中innerHTML用法实例分析
2015/01/12 Javascript
JS更改select内option属性的方法
2015/10/14 Javascript
全面解析多种Bootstrap图片轮播效果
2016/05/27 Javascript
javascript js 操作数组 增删改查的简单实现
2016/06/20 Javascript
全面接触神奇的Bootstrap导航条实战篇
2016/08/01 Javascript
jquery自定义插件结合baiduTemplate.js实现异步刷新(附源码)
2016/12/22 Javascript
vue使用drag与drop实现拖拽的示例代码
2017/09/07 Javascript
Vue中插入HTML代码的方法
2018/09/21 Javascript
如何利用vue+vue-router+elementUI实现简易通讯录
2019/05/13 Javascript
vue+element-ui+axios实现图片上传
2019/08/20 Javascript
JS前后端实现身份证号验证代码解析
2020/07/23 Javascript
Python实现短网址ShortUrl的Hash运算实例讲解
2015/08/10 Python
详解Python的Twisted框架中reactor事件管理器的用法
2016/05/25 Python
ubuntu环境下python虚拟环境的安装过程
2018/01/07 Python
python 实现对文件夹内的文件排序编号
2018/04/12 Python
PyQt5每天必学之弹出消息框
2018/04/19 Python
Python实现的批量修改文件后缀名操作示例
2018/12/07 Python
python使用SQLAlchemy操作MySQL
2020/01/02 Python
python 对任意数据和曲线进行拟合并求出函数表达式的三种解决方案
2020/02/18 Python
Python模拟FTP文件服务器的操作方法
2020/02/18 Python
解决import tensorflow as tf 出错的原因
2020/04/16 Python
matlab、python中矩阵的互相导入导出方式
2020/06/01 Python
Python如何读写CSV文件
2020/08/13 Python
const和static readonly区别
2013/05/20 面试题
研究生自荐信
2013/10/09 职场文书
法人授权委托书样本
2014/09/19 职场文书
产品调价通知函
2015/04/20 职场文书
交通事故调解协议书
2015/05/20 职场文书
感谢师恩主题班会
2015/08/17 职场文书
志愿者工作心得体会
2016/01/15 职场文书
OpenCV-Python实现油画效果的实例
2021/06/08 Python