解决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 相关文章推荐
用js实现的检测浏览器和系统的函数
Apr 09 Javascript
Javascript结合css实现网页换肤功能
Nov 02 Javascript
jQuery 操作下拉列表框实现代码
Feb 22 Javascript
轻量级 JS ToolTip提示效果
Jul 20 Javascript
javascript for循环从入门到偏门(效率优化+奇特用法)
Aug 01 Javascript
JQuery事件e参数的方法preventDefault()取消默认行为
Sep 26 Javascript
vue.js之vue-cli脚手架的搭建详解
May 05 Javascript
详解AngularJS用Interceptors来统一处理HTTP请求和响应
Jun 08 Javascript
vue better-scroll插件使用详解
Jan 25 Javascript
node中间层实现文件上传功能
Jun 11 Javascript
JavaScript 扩展运算符用法实例小结【基于ES6】
Jun 17 Javascript
JS绘图Flot如何实现可选显示曲线图功能
Oct 16 Javascript
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 获取本机外网/公网IP的代码
2010/05/09 PHP
php中simplexml_load_file函数用法实例
2014/11/12 PHP
php 二维数组时间排序实现代码
2016/11/19 PHP
PHPStudy下如何为Apache安装SSL证书的方法步骤
2019/01/23 PHP
Mootools 1.2教程 Tooltips
2009/09/15 Javascript
JavaScript起点(严格模式深度了解)
2013/01/28 Javascript
FireBug 调试JS入门教程 如何调试JS
2013/12/23 Javascript
简单介绍JavaScript的变量和数据类型
2015/06/03 Javascript
深入解析JavaScript中的数字对象与字符串对象
2015/10/21 Javascript
jQuery解析json数据实例分析
2015/11/24 Javascript
js实现文字闪烁特效的方法
2015/12/17 Javascript
js实现3D图片环展示效果
2017/03/09 Javascript
基于JavaScript实现图片连播和联级菜单实例代码
2017/07/28 Javascript
用js屏蔽被http劫持的浮动广告实现方法
2017/08/10 Javascript
[02:44]2014DOTA2 国际邀请赛中国区预选赛 大神红毯秀
2014/05/25 DOTA
python采集博客中上传的QQ截图文件
2014/07/18 Python
Python入门学习之字符串与比较运算符
2015/10/12 Python
python查询mysql,返回json的实例
2018/03/26 Python
Python中按键来获取指定的值
2019/03/02 Python
Python如何应用cx_Oracle获取oracle中的clob字段问题
2019/08/27 Python
python中的函数递归和迭代原理解析
2019/11/14 Python
浅谈tensorflow模型保存为pb的各种姿势
2020/05/25 Python
python属于解释语言吗
2020/06/11 Python
python使用matplotlib的savefig保存时图片保存不完整的问题
2021/01/08 Python
中国领先的专业家电网购平台:国美在线
2016/12/25 全球购物
沪江旗下的海量优质课程平台:沪江网校
2017/11/07 全球购物
夏威夷灵感服装及配饰:Reyn Spooner
2018/09/18 全球购物
美国购买新书和二手书网站:Better World Books
2018/10/31 全球购物
会计自荐书
2013/12/02 职场文书
学校大课间活动方案
2014/01/30 职场文书
小学英语教师先进事迹
2014/05/28 职场文书
医生个人自我剖析材料
2014/10/08 职场文书
教师求职简历自我评价
2015/03/10 职场文书
2015年材料员工作总结
2015/04/30 职场文书
好员工观后感
2015/06/17 职场文书
CSS3鼠标悬浮过渡缩放效果
2021/04/17 HTML / CSS