【面试题】如何在JS中控制任务并发数?

前言

最近看到很多面经分享中都出现过这么一道题目

实现一个类或函数,使其支持最大并发数为2

这个问题看上去非常实用啊,如果说大型应用中需要批量发起大量的请求,浏览器单域名http1.1 以下最多只能共存6条TCP连接,有可能会阻塞很多重要的任务,比如下载或者上传,你需要控制并发数量,就需要自行实现一个任务调度器

思考

  1. 实现这么一个东西肯定是需要一个任务队列的
  2. 这个队列只能同时取出n个任务来执行
  3. 当任务结束后立即取下一个任务开始执行,直到任务队列为空

实现

按照上面的思路,我们先设计这个对象

class TaskScheduler {
    constructor(concurrentCount = 2) {
        // 并发上限
        this.concurrentCount = concurrentCount;
        // 运行中的任务数
        this.runningTaskCount = 0;
        // 任务列表
        this.tasks = [];
    }
    
    addTask(task) {
        return new Promise((resolve, reject) => {
            // 添加任务到队列里
            this.tasks.push({
                task,
                resolve,
                reject,
            });
        });
    }
}

到这里,大家可能会发现,任务加进去了,怎么执行呢?

添加的方法只管添加任务,我们还需要一个 run 方法来做任务的调度

class TaskScheduler {
    constructor(concurrentCount = 2) {
        // 并发上限
        this.concurrentCount = concurrentCount;
        // 运行中的任务数
        this.runningTaskCount = 0;
        // 任务列表
        this.tasks = [];
    }
    
    addTask(task) {
        return new Promise((resolve, reject) => {
            // 添加任务到队列里
            this.tasks.push({
                task,
                resolve,
                reject,
            });
            // 添加任务之后立即进入执行
            this._run();
        });
    }
    
    _run() {
        // 当任务列表不为空 且 正在运行的任务不超过并发上限 则继续执行下一个任务
        while (this.tasks.length > 0 && this.runningTaskCount < this.concurrentCount) {
            // 队列拿最新的任务
            const { task, resolve, reject } = this.tasks.shift();
            
            // 运行计数 + 1
            this.runningTaskCount++;
            
            // 运行任务
            const res = task();
            
            // 判断任务是异步还是同步任务
            if (res instanceof Promise) {
                res.then(resolve, reject).finally(() => {
                    // 执行结束 运行计数 - 1
                    this.runningTaskCount--;
                    // 递归调用 run 执行下一个任务
                    this._run();
                });
            } else {
                // 执行结束 运行计数 - 1
                this.runningTaskCount--;
                // 递归调用 run 执行下一个任务
                this._run();
            }
        }
    }

到这里,一个简单的并发任务调度器已经实现了,其实代码并不多,而且也不是很复杂,基本是围绕上面的几个条件来实现的

这里可能会有小伙伴纠结 _run() 函数中为什么需要用 while 是不是用 if 可以达到同样的效果?

的确,如果任务都是由 addTask 来进行添加的话,的确不需要用到 while,因为一添加就直接执行了,而且任务结束也会递归调用

但如果一开始就有一个初始化的任务列表的话,那么 if 就不够用了,就需要用到 while 了,代码如下

class TaskScheduler {
    constructor(concurrentCount = 2, tasks = []) {
        ...
        // 任务列表
        this.tasks = tasks;
    }
    
    addTask(task) {
        ...
    }
    
    _run() {
        ...
    }

不然就只能等第一个结束之后才能执行下一个任务,就变成并发只有 1 的调度器了

这里提供一个测试用例供大家测试:
GitHub 代码完整版+测试用例

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容