JavaScript 任务池
JavaScript 任务池
本文写于 2022 年 5 月 13 日
线程池
在多线程语言中,我们通常不会随意的在需要启动线程的时候去启动,而是会选择创建一个线程池。
所谓线程池,本意其实就是(不止这些作用,其余作用可以自行查阅):
- 节省操作系统资源
- 限制最大线程数。
对于 JavaScript 来说,虽然不存在“启动线程”这种问题,但我们还是可以通过类似的思想,来限制我们做异步操作的数量。
分析
首先我们需要一个数组,用它来存储尚未执行的任务,每个任务都是一个函数,这个函数必须要返回一个 Promise。
type Task = () => Promise<unknown>;
const tasks: Task[] = [];
其次我们需要一个方法来进行任务的添加。
function addTask(task: Task): void;
最后我们需要一个函数来执行我们所有的 task。
而在这之前,我们还需要定义一个值,来定义同时执行的异步任务的最大数量。
function execTasks(): void;
实现
根据我们的分析,我们可以写下基础的代码如下:
interface TaskPool {
addTask(task: Task): void;
}
type Task = () => Promise<unknown>;
function newTaskPool(max = 10): TaskPool {
const tasks: Task[] = [];
function addTask(task: Task): void {}
function execTasks(): void {}
}
新增任务非常简单,我们写出如下代码填充 addTask。
function addTask(task: Task): void {
tasks.push(task);
}
接下来就是重头戏。如何实现 execTasks 方法来限制最大异步任务数量呢?
首先我们来明确一点,在下面这个场景中,如果 foo 函数是异步操作,那么是不会阻塞我们的代码执行的。
console.log("Before");
foo();
console.log("After");
那么我们可以这么操作:
- 定义一个变量用来记录当前的空闲任务数量;
- 执行
execTasks时,会选取当前任务数量和空闲任务数二者相比较小的一个; - 根据该值进行循环,每次循环弹出
tasks第一位的任务进行执行; - 执行前将空闲任务数 -1,执行完毕后空闲任务数 +1,并再次执行
execTasks。
let leisure = max;
function execTasks(): void {
if (tasks.length === 0) return;
const execTaskNum = Math.min(tasks.length, leisure);
for (let i = 0; i < execTaskNum; i++) {
const task = tasks.shift();
if (!task) continue;
leisure--;
task().finally(() => {
leisure++;
execTasks();
});
}
}
最后我们只剩下了一个问题了,我们如何在 addTask 后执行 execTasks,但又不会让下面这种情况导致频繁执行 execTasks:
for (let i = 0; i < 100; i++) addTask();
可以利用防抖 + setTimeout(() => {},0) 的特性来完成。
function addTask(task: Task) {
tasks.push(task);
execTasksAfterAdd();
}
// 这里借用了 lodash 的 debounce 函数,具体实现不多说,可以看我以前的文章:防抖与节流
const execTasksAfterAdd = debounce(execTasks);
完整代码:
import { debounce } from "lodash";
interface TaskQueue {
addTask: (task: () => Promise<any>) => void;
}
function newTaskQueue(maxTaskNum = 10): TaskQueue {
let _leisure = maxTaskNum;
const _tasks: Array<() => Promise<any>> = [];
function addTask(task: () => Promise<any>) {
_tasks.push(task);
execAfterTask();
}
const execAfterTask = debounce(execTasks);
function execTasks() {
if (_tasks.length === 0) return;
const execTaskNum = Math.min(_tasks.length, _leisure);
for (let i = 0; i < execTaskNum; i++) {
const task = _tasks.shift();
if (!task) continue;
_leisure--;
task().finally(() => {
_leisure++;
execTasks();
});
}
}
return { addTask };
}
const queue = newTaskQueue(5);
for (let i = 0; i < 10; i++) {
queue.addTask(function () {
return new Promise<void>((resolve) => {
setTimeout(() => resolve(), 800);
});
});
}
使用场景
其实这种做法的使用场景是比较少的。
绝大多数情况我们都不需要这么去做,除非碰到很极端的需求。
例如我们需要用 Node.js 去设计一个吞吐量极大的服务,那么同时发生大量的网络请求很可能把带宽直接打满,导致后续的请求无法打到该服务,此时就可以使用任务池来控制最大网络请求量。
(完)
JavaScript 任务池的更多相关文章
- javascript中的队列结构
1.概念 队列和栈结构不同,栈是一种后进先出的结构,而队列是一种先进先出的结构.队列也是一种表结构,不同的是队列只能在队尾插入元素,在队首删除元素,可以将队列想象成一个在超时等待排队付钱的队伍,或者在 ...
- 第五章:javascript:队列
队列是一种列表,不同的是队列只能在末尾插入元素,在队首删除元素.队列用于存储按顺序排列的数据.先进先出.这点和栈不一样,在栈中,最后入栈的元素反被优先处理.可以将队列想象成银行排队办理业务的人,排队在 ...
- 数据结构与算法JavaScript描述——队列
注:澄清一个bug: /** * 删除队首的元素: */ function dequeue(){ return this.dataStore.shift(); } 应该有return: 队列是一种 ...
- JavaScript之父Brendan Eich,Clojure 创建者Rich Hickey,Python创建者Van Rossum等编程大牛对程序员的职业建议
软件开发是现时很火的职业.据美国劳动局发布的一项统计数据显示,从2014年至2024年,美国就业市场对开发人员的需求量将增长17%,而这个增长率比起所有职业的平均需求量高出了7%.很多人年轻人会选择编 ...
- javascript中的Array对象 —— 数组的合并、转换、迭代、排序、堆栈
Array 是javascript中经常用到的数据类型.javascript 的数组其他语言中数组的最大的区别是其每个数组项都可以保存任何类型的数据.本文主要讨论javascript中数组的声明.转换 ...
- Javascript 的执行环境(execution context)和作用域(scope)及垃圾回收
执行环境有全局执行环境和函数执行环境之分,每次进入一个新执行环境,都会创建一个搜索变量和函数的作用域链.函数的局部环境不仅有权访问函数作用于中的变量,而且可以访问其外部环境,直到全局环境.全局执行环境 ...
- 探究javascript对象和数组的异同,及函数变量缓存技巧
javascript中最经典也最受非议的一句话就是:javascript中一切皆是对象.这篇重点要提到的,就是任何jser都不陌生的Object和Array. 有段时间曾经很诧异,到底两种数据类型用来 ...
- 读书笔记:JavaScript DOM 编程艺术(第二版)
读完还是能学到很多的基础知识,这里记录下,方便回顾与及时查阅. 内容也有自己的一些补充. JavaScript DOM 编程艺术(第二版) 1.JavaScript简史 JavaScript由Nets ...
- 《Web 前端面试指南》1、JavaScript 闭包深入浅出
闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...
随机推荐
- 学习GlusterFS(八)
GlusterFS集群创建 一.简介 GlusterFS概述 Glusterfs是一个开源的分布式文件系统,是Scale存储的核心,能够处理千数量级的客户端.在传统的解决 方案中Glusterfs能够 ...
- 学习GlusterFS(四)
基于 GlusterFS 实现 Docker 集群的分布式存储 以 Docker 为代表的容器技术在云计算领域正扮演着越来越重要的角色,甚至一度被认为是虚拟化技术的替代品.企业级的容器应用常常需要将重 ...
- 2.安装Spark与Python练习
一.安装Spark <Spark2.4.0入门:Spark的安装和使用> 博客地址:http://dblab.xmu.edu.cn/blog/1307-2/ 1.1 基础环境 1.1.1 ...
- BMZCTF 端午节就该吃粽子
端午节就该吃粽子 题目如下让我们访问login.php 然后就一个登录界面查看源码发现index.php 我们直接访问发现没有结果使用伪协议读取 然后我们使用base64解密 <?php err ...
- 纯干货数学推导_傅里叶级数与傅里叶变换_Part5_从傅里叶级数推导傅里叶变换
- java中如果我老是少捕获什么异常,如何处理?
马克-to-win:程序又一次在出现问题的情况下,优雅结束了.上例中蓝色部分是多重捕获catch.马克-to-win:观察上面三个例子,结论就是即使你已经捕获了很多异常,但是假如你还是少捕获了什么异常 ...
- java中抽象类和抽象方法到底什么关系?请举例说明!
抽象类和抽象方法什么关系?抽象类中可能有抽象方法,也可能没有抽象方法.那位说,就跟没说一样,那抽象类和抽象方法都叫抽象,他们必定有关系,那关系是什么呢?如果一个类中有抽象方法,它必须得是抽象类. An ...
- 牛客网-剑指Offer 二维数组中的查找
题目描述 在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数 ...
- nginx之配置文件公用抽取
nginx之配置文件公用抽取 因为某些原因,需要同时部署同一应用两个不同分支的代码,而配置文件存在较大重复,因此有此篇. 最近构建的过程中遇到了一些跟nginx配置相关的问题,记录下. 简单说下构建的 ...
- javaScript设计模式:发布订阅模式
发布订阅模式的思想是在观察者模式的基础上演变而来,在观察者模式中客户端监听到对象某个行为就触发对应任务程序.而在发布订阅模式中依然基于这个核心思想,所以有时候也会将两者认为是同一种设计模式.它们的不同 ...