如何突破ajax并发的限制,利用js实现Ajax并发请求限制请求数量的示例代码

出现问题描述:当不确定异步请求个数时,为防止当一瞬间发生上百个http请求时,导致堆积了无数调用栈进而导致内存溢出问题。

要求:将同一时刻并发请求数量控制在3个以内,同时还要尽可能快速的拿到响应的结果。

同面试问题:

实现一个批量请求函数 multiRequest(urls, maxNum),要求如下:

要求最大并发数 maxNum

每当有一个请求返回,就留下一个空位,可以增加新的请求

所有请求完成后,结果按照 urls 里面的顺序依次打出

1、基于Promise.all实现Ajax的串行和并行

平时都是基于promise来封装异步请求的

串行:一个异步请求完成了之后再进行下一个请求

并行:多个异步请求同时进行

示例:串行

var p = function () {

return new Promise(function (resolve, reject) {

setTimeout(() => {

console.log('1000')

resolve()

}, 1000)

})

}

var p1 = function () {

return new Promise(function (resolve, reject) {

setTimeout(() => {

console.log('2000')

resolve()

}, 2000)

})

}

var p2 = function () {

return new Promise(function (resolve, reject) {

setTimeout(() => {

console.log('3000')

resolve()

}, 3000)

})

}

p().then(() => {

return p1()

}).then(() => {

return p2()

}).then(() => {

console.log('end')

})

并行:

var promises = function () {

return [1000, 2000, 3000].map(current => {

return new Promise(function (resolve, reject) {

setTimeout(() => {

console.log(current)

}, current)

})

})

}

Promise.all(promises()).then(() => {

console.log('end')

})

Promise.all(promises: []).then(fun: function);

promise.all保证数组中所有promise对象都达到resolve状态,才执行then回调

Promise.all并发限制

含义: 指每个时刻并发执行的promise数量是固定的,最终执行的结果还是保持与原来的promise.all一致。

思路与实现

采用递归调用来实现,设置最大请求数量上限。并在这些请求中的每一个都应该在完成时继续递归发送,通过传入的索引来确定了urls里面具体是那个URL,保证最后输出的顺序不会乱,而是依次输出

代码实现:

function multiRequest(urls = [], maxNum) {

// 请求总数量

const len = urls.length;

// 根据请求数量创建一个数组来保存请求的结果

const result = new Array(len).fill(false);

// 当前完成的数量

let count = 0;

return new Promise((resolve, reject) => {

// 请求maxNum个

while (count < maxNum) {

next();

}

function next() {

let current = count++;

// 处理边界条件

if (current >= len) {

// 请求全部完成就将promise置为成功状态, 然后将result作为promise值返回

!result.includes(false) && resolve(result);

return;

}

const url = urls[current];

console.log(`开始 ${current}`, new Date().toLocaleString());

fetch(url)

.then((res) => {

// 保存请求结果

result[current] = res;

console.log(`完成 ${current}`, new Date().toLocaleString());

// 请求没有全部完成, 就递归

if (current < len) {

next();

}

})

.catch((err) => {

console.log(`结束 ${current}`, new Date().toLocaleString());

result[current] = err;

// 请求没有全部完成, 就递归

if (current < len) {

next();

}

});

}

});

}

代码实现:

// 任务列表->新建任务

uploadFile() {

let _this = this;

var uploadThreadLimitNums = 3,

uploadThreadNums = 0,

sendFinishNum = 0,

resultFinishNum = 0;

var marks = 0;

var tasks = [];

var upload = function () {

while (uploadThreadNums < uploadThreadLimitNums) {

if (sendFinishNum >= _this.fileList.length) {

if (resultFinishNum >= _this.fileList.length) {

creatTask(); // 完成请求

}

return;

}

(function (j) {

let item = _this.fileList[j];

let p = new FormData();

p.append("file", item);

tasks.push(

axios({

method: "post",

url: `${window.UL_CONFIG.BASEURL}/api/files/upload`,

data: p,

onUploadProgress: (progressEvent) => {

for (let i in _this.rowData) {

_this.rowData[i].name === item.name

? (_this.rowData[i].percent = Math.round(

(progressEvent.loaded / progressEvent.total) * 100

))

: "";

}

},

})

.then((res) => {

/* let obj = {};

obj.url = `${window.UL_CONFIG.BASEURL}/api/files/${res.data}`;

obj.fileName = item.name;

obj.fmt = _this.ruleForm.format;

obj.samplingRate = _this.ruleForm.samplingRate;

fileUrls.push(obj); */

})

.catch((e) => {

? (_this.rowData[i].percent = 0)

_this.$notify.error({

title: "错误",

message: "服务连接错误 " + item.name + " 未上传成功",

});

.finally(() => {

uploadThreadNums--;

resultFinishNum++;

upload();

);

})(sendFinishNum);

uploadThreadNums++;

sendFinishNum++;

}

};

var creatTask = function () {

axios.all(tasks).then((res) => {

// 新建上传任务

/* let fd1, fd2, calcFlag, flagArr, language;

fd1 = {};

flagArr = Object.assign([], _this.ruleForm.checkList);

if (_this.ruleForm.recognize == "自动识别") {

flagArr.push("2");

calcFlag = flagArr.reduce(

(accu, curr) => Number(accu) + Number(curr)

);

_this.ruleForm.recognize == "自动识别"

? (language = "")

: (language = _this.ruleForm.recognize);

fd1.processContent = calcFlag;

fd1.remark = _this.ruleForm.remark;

fd1.name = _this.ruleForm.taskName;

fd1.fmt = _this.ruleForm.format;

fd1.samplingRate = _this.ruleForm.samplingRate;

fd1.language = language;

fd1.type = 1; // type: 1 语音, 2 视频

fd1.files = fileUrls; */

newTask(fd1).then((res) => {

/* _this.cmpltBtnState = false;

_this.$store.commit("setTaskId", res.data.id);

_this.submitFailNumber = res.data.submitFailNumber; */

_this.$parent.dataInit();

});

});

upload();

},

到此这篇关于利用js实现Ajax并发请求限制请求数量的示例代码的文章就介绍到这了,更多相关js Ajax并发请求限制内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:Jquery中$.get(),$.post(),$.ajax(),$.getJSON()的用法总结

JQuery处理json与ajax返回JSON实例代码

$.ajax json数据传递方法

jquery的ajax异步请求接收返回json数据实例

jQuery Ajax异步处理Json数据详解

原生js实现ajax方法(超简单)

jquery的ajax和getJson跨域获取json数据的实现方法

javascript实现原生ajax的几种方法介绍

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值