js中的多并发处理。

通常 为了减少页面加载时间,先把核心内容显示处理,页面加载完成后再发送ajax请求获取其他数据

这时就可能产生多个ajax请求,为了用户体验,最好是发送并行请求,这就产生了并发问题,应该如何处理?

(1)并行改串行

如果业务逻辑和用户体验允许的情况下,可以改为串行,处理起来最简单

function async1(){
//do sth...
async2();
}
function async2(){
//do sth...
}
async1();

(2)回调计数

function async1(){
//do sth...
callback();
}
function async2(){
//do sth...
callback();
}
function callback(){
cnt++;
if(2==cnt) console.log('都已执行完毕');
}

(3)循环非阻塞

function async1(){
//do sth...
cnt++;
}
function async2(){
//do sth...
cnt++;
}
var interval = setInterval(function(){
if(2===cnt){
console.log('已执行完成');
clearInterval(interval)
}
};

这个方法采用了定时间隔触发器,占用CPU比较多,建议酌情使用

(4)jquery

使用jquery的延时处理方法,每个ajax请求完成后,把对应的Deferred置为完成状态,然后用jquery判断全部完成后再进行后续处理

var d1 = $.Deferred();
var d2 = $.Deferred();

function async1(){
d1.resolve( "Fish" );
}

function async2(){
d2.resolve( "Pizza" );
}

$.when( d1, d2 ).done(function ( v1, v2 ) {
console.log( v1 + v2 + '已完成');
});

另:ajax时异步请求不会出现堵塞情况。

$.when($.ajax, $ajax).done(function() { balabala })

转载:

经常在写代码的时候碰到这样的场景:页面初始化时显示loading页,同时启动多个ajax并发请求获取数据,当每个ajax请求返回时结束loading。 
举个例子,一个下订单的页面,要查询常用地址信息、商品信息、地市信息…而这些请求都是异步的,希望等到所有数据加载完成后再允许用户操作。 
要实现这个场景容易碰到的一个问题就是多并发怎么控制?下面是一些解决方法和思路:

并行改为串行 
如果业务逻辑本身是串行的,但是提供的请求方式又是异步的,可以考虑此方法。 
但本场景显然不是这种情况,这样做大大降低了页面性能,延长了加载速度。

回调 
只适合并发数少的情况,多层嵌套回调会让代码的可读性大大降低

  1. function async1(){
  2. //do sth...
  3. }
  4. function async2(){
  5. //do sth...
  6. async1();
  7. }
  8. async2();

ajax改为同步

如在jquery中将async参数设置为false。

  1. $.ajax({
  2. url:"/jquery/test1.txt",
  3. async:false
  4. });

设置结束标识 
简单一点的可以设置计数器,每完成一个异步函数加1,或者设置一个数组,每执行完一个异步函数更新数组。

回调计数

  1. var cnt = 0;
  2. function async1(){
  3. //do sth...
  4. callback();
  5. }
  6. function async2(){
  7. //do sth...
  8. callback();
  9. }
  10. function callback(){
  11. cnt++;
  12. if(2==cnt) console.log('都已执行完毕');
  13. }

循环阻塞

  1. var cnt = 0;
  2. function async1(){
  3. //do sth...
  4. cnt++;
  5. }
  6. function async2(){
  7. //do sth...
  8. cnt++;
  9. }
  10. while(2>cnt){}

循环非阻塞

不建议过多使用,以免影响性能

  1. var cnt = 0;
  2. function async1(){
  3. //do sth...
  4. cnt++;
  5. }
  6. function async2(){
  7. //do sth...
  8. cnt++;
  9. }
  10. var interval = setInterval(function(){
  11. if(2===cnt){
  12. console.log('已执行完成');
  13. clearInterval(interval)
  14. }
  15. }, 0);

第三方框架实现 
jquery

目前我在项目中采用的方式

    1. var d1 = $.Deferred();
    2. var d2 = $.Deferred();
    3. function async1(){
    4. d1.resolve( "Fish" );
    5. }
    6. function async2(){
    7. d2.resolve( "Pizza" );
    8. }
    9. $.when( d1, d2 ).done(function ( v1, v2 ) {
    10. console.log( v1 + v2 + '已完成');
    11. });

JS中如何处理多个ajax并发请求?的更多相关文章

  1. MVC中用View.bag保存json字符串。在js中使用,不用ajax请求。。。。

    有时候我们只需要使用一次json数据,这时候直接在后台查出json,然后用view.bag传到前台使用,就会很方便..(在前台用ajax请求的话,感觉有点多余..) 上代码 后台: public Ac ...

  2. springMVC中使用 RequestBody 及 Ajax POST请求 415 (Unsupported Media Type)

    使用POST请求的时候一直报错: Ajax 未设置 contentType 时会报 415 . 后台 RequestBody  承接前台参数,故对参数data的要求为“必传”“JSON”,否则会报40 ...

  3. js中使用队列发送ajax请求

    最近,项目中需要按照先后顺序发送ajax请求,并且在一次请求结束后才能发起下一次,不然就会导致逻辑错误. 解决办法是定义一个数组,保存ajax请求数据. 以下使用extjs4定义一个类 Ext.def ...

  4. 2.jquery在js中写标准的ajax请求

    $(function(){ $.ajax({ url:"http://www.microsoft.com", //请求的url地址 dataType:"json" ...

  5. 关于.Net Core 前后端分离跨域请求时 ajax并发请求导致部分无法通过验证解决办法。

    项目中有这样一个页面.页面加载的时候会同时并发6个ajax请求去后端请求下拉框. 这样会导致每次都有1~2个“浏览器预请求”不通过. 浏览器为什么会自动发送“预请求”?请看以面连接 https://b ...

  6. CI框架中,判断post,ajax,get请求的方法

    CI框架当中并没有提供,类似tp框架中IS_POST,IS_AJAX,IS_GET的方法. 所有就得我们自己造轮子了.下面就介绍一下,如何定义这些判断请求的方法.其实很简单的. 首先打开constan ...

  7. js中如何处理大量有规律的变量

    var a1=document.getElementById('a1'); var a1=document.getElementById('a2'); var a1=document.getEleme ...

  8. JavaScript异步并发请求问题

    JavaScript异步并发请求问题 JS中如何处理多个ajax并发请求? jQuery的deferred对象详解 面试遇到的ajax请求串行和并行问题

  9. springMVC框架在js中使用window.location.href请求url时IE不兼容问题解决

    是使用springMVC框架时,有时候需要在js中使用window.location.href来请求url,比如下面的路径: window.location.href = 'forecast/down ...

随机推荐

  1. bzoj2565 最长双回文子串

    Description 顺序和逆序读起来完全一样的串叫做回文串.比如acbca是回文串,而abc不是(abc的顺序为“abc”,逆序为“cba”,不相同).输入长度为n的串S,求S的最长双回文子串T, ...

  2. 正则捕获的细节及replace分析

    1.var reg=/./与var reg=/\./的区别? 前者代表任意一个字符, 后者代表这个字符串中得有一个点 2.?的使用 如果单独的一个字符后面带? 代表1个或0个这个字符的出现: 列如: ...

  3. [UE4]通过代码改变材质

    OrangeMaterial = ConstructorStatics.OrangeMaterial.Get(); , OrangeMaterial); 使用到的结构体如下: struct FCons ...

  4. PHP-Socket服务端客户端发送接收通信实例详解

    原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://fighter.blog.51cto.com/1318618/1533957 So ...

  5. android官方文档翻译(不断更新中。。。)

    最近在自学android,抽空把官方文档的guide跟training差不多看了一遍,又对比了一些书籍,感觉还是官方文档讲得比较好,所以自己计划把官方文档翻译一下,方便自己的知识巩固以及复习查找,由于 ...

  6. 使用CMQ和SCF实现邮件发送

    准备腾讯云 API 调用工具 使用 API 命令行工具来管理和运行无服务器云函数(SCF),下面就先来安装配置该工具. 安装 Python 和 PIP Python 环境是腾讯云命令行工具运行时的必要 ...

  7. MONGODB用户、角色和权限管理

      最近遇到很多问MONGODB用户认证授权方面的问题,现在特记录下来,与大家共享. 一.概念理解 1.用户概念 Mongodb的用户是由 用户名+所属库名组成 例如: 登录mongo  testdb ...

  8. 第一个python抓取单网页的例子

    #!/usr/bin/env python # coding=utf-8 import requests from bs4 import BeautifulSoup import pymysql im ...

  9. JAVA Serverlet 请求头信息和响应头信息

    <1>获取请求头信息 //获取请求头信息的全部名称 Enumeration<String> header = request.getHeaderNames(); while(h ...

  10. oracle跟踪sql语句

    oracle跟踪sql语句 select * from v$sql 查询客户端电脑名称的ID select terminal, SID,SERIAL#  from v$session where  ( ...