在我们平时开发中,经常会遇到页面数据初始化时,频繁调同一个接口的情况。比如echarts项目中,一个页面可能会有几十张图表,如果一个接口返回所有图表数据的话,会造成用户过长的等待时间,再者过多图表同时渲染,也会给页面增加压力,造成卡顿的现象。
我们通常会让每个图表单独调一个接口,入参不同,这样更有利于页面快速渲染图表,单个图表请求到数据,立即渲染,不需要等待其他图表。可理想很丰满,现实很骨感,当服务器配置过低,或者后端代码性能较弱,会难以处理这些并发请求,接口调用越多,等待处理的时间可能就越长,甚至超过一次性返回所有数据的间。。。为了解决这种问题,缓解后端压力,本篇将介绍前端来控制请求的并发数:
先分析一波,假设我们需要重复调用30次接口,并联调用接口,服务端压力较大,可能会造成响应时间过长。逐渐减少并发数,假设并发数为5的时候,服务器处理速度最快,几乎不受并发影响。
针对这种情况,我们可以封装接口请求方法,控制每次接口请求的并发数,将30次分解成:并发数为5,分6次请求。这样的话,服务器每次处理5次请求,资源释放出来继续处理下一批请求,从而解决并发拥堵问题~
初步构思:
class TaskQueue {
constructor(max) {
this.max = max;
this.taskList = [];
}
addTask(task) {
this.taskList.push(task);
}
}
function createTask(i) {
return () => {
return new Promise((resolve, reject) => {
setTimeout(() => {
if (i == 4 || i == 15) {
reject("出错啦~");
} else {
resolve("成功呀~" + i);
}
}, 2000);
});
};
}
const taskQueue = new TaskQueue(5);
for (let i = 0; i < 30; i++) {
const task = createTask(i);
taskQueue.addTask(task);
}
for循环调用函数createTask()返回30个promise的异步任务,任务队列TaskQueue类返回一个实例,控制这30个异步任务的并发,构造器中传入并发数5。
接下来用TaskQueue实现控制并发:
class TaskQueue {
constructor(max) {
this.max = max; // 并发数
this.min = 0;
this.taskList = []; // 全部任务
Promise.resolve().then(() => this.run()) // 等同步代码(addTask)全部执行完成,再执行run
}
// 增加任务
addTask(task) {
this.taskList.push(task);
}
// 执行任务
async run() {
if (!this.taskList.length) return;
const AsyncTasks = [];
this.min = Math.min(this.max, this.taskList.length) // 当传入的并发数大于任务数,取任务数, 反之取并发数
// 根据并发数分组
for(let i = 0; i < this.min; i++) {
AsyncTasks.push(this.taskList.shift());
}
await this.handleTask(AsyncTasks); // 通过下面递归,这里将会有6个异步任务串联执行
this.run(); // 递归
}
async handleTask(tasks) {
// 返回promise处理异步任务组
return new Promise(resolve => {
// 遍历任务组,5个异步任务并联执行
tasks.forEach(async (task, index) => {
await task().then(res => {
console.log(res);
}).catch((err) => {
console.log(err);
}).finally(() => {
index + 1 === this.min && console.log('===============================');
index + 1 === this.min && resolve() // 最后一个任务resolve(),promise完成
})
})
})
}
}
function createTask(i) {
return () => {
return new Promise((resolve, reject) => {
setTimeout(() => {
if (i == 4 || i == 15) { // 测试捕捉错误
reject("出错啦~");
} else {
resolve("成功呀~" + i);
}
}, 2000);
});
};
}
const taskQueue = new TaskQueue(5);
for (let i = 0; i < 30; i++) {
const task = createTask(i);
taskQueue.addTask(task);
}
试试效果:
nice,至此,30次异步任务,分6次完成,每次处理5个,大家可以在此基础上拓展请求接口,并增加一些处理逻辑,欢迎留言探讨~
来自:https://www.cnblogs.com/coder--wang/archive/2022/08/01/16540998.html
强制把http请求跳转到https,结果发现App有部分的功能不能使用,因为App一共设置了4种请求方式,分别是GET,POST,DELETE和OPTIONS方式,设置301跳转后所有的请求方法都变成了GET方式,导致一些功能无法正常使用.
几乎所以有关Nginx书只要是讲深入点的就会讲到Nginx请求的11个处理阶段,要记住这些真是不易,人脑特别不擅长记住各种东西,只能做些索引罢了,能做到知道这个知识点在哪儿能找到不就行了,可是你去面试还是问这些理论,所以这里汇总下记录如下
HTTP通信机制是在一次完整的HTTP通信过程中,Web浏览器与Web服务器之间将完成下列7个步骤:建立TCP连接、Web浏览器向Web服务器发送请求命令、Web浏览器发送请求头信息、 Web服务器应答
http请求中的8种请求方法:opions 返回服务器针对特定资源所支持的HTML请求方法 ,Get 向特定资源发出请求,Post 向指定资源提交数据进行处理请求
通过node提供的http模块,可以通过其提供的get()和request()两个方法发起http请求,get()是对request()方法的封装,方便发起get请求,如果要实现post请求,那么需要对request()方法进行封装。
遇到这样一种情况,打开网页两个窗口a,b(都是已经登录授权的),在a页面中退出登录,然后在b页面执行增删改查,这个时候因为授权原因,b页面后端的请求肯定出现异常(对这个异常的处理,进行内部跳转处理),b页面中的ajax请求的回调中就会出现问题
本文所讲的 POST 请求是 HTTP/1.1 协议中规定的众多 HTTP 请求方法的其中最常用的一个。一般使用 POST 请求方法向服务器发送数据(主要是一些创建更新操作),本文讨论的是 POST 请求方法常用的四种数据提交格式。
flutter一直很火的网络请求插件dio,直接上代码,写成一个类,可以直接使用,包含请求的封装,拦截器的封装
当我们在web浏览器的地址栏中输入: www.baidu.com,然后回车,到底发生了什么?DNS域名解析采用的是递归查询的方式,过程是,先去找DNS缓存->缓存找不到就去找根域名服务器->根域名又会去找下一级
nginx首先决定要用配置文件里的哪个server{}块来处理,假设有下面的server{}配置;nginx会根据过来的http请求头里的Host字段里的值,来判断使用哪个server{}。
内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!