【JS】实现 Promise 的并发数控制

前言

首先 JS 自带两个并发控制的函数 Promise.allPromise.race,前者用来实现并发运行参数列表中的所有 Promise;后者同样并发运行所有 Promise,但是当任意一个 Promise 运行完成函数就会返回:

async function myFetch() {
  return new Promise(resolve => {
  	setTimeout(resolve, 1000)
  })
}

// myFetch 并发执行,返回参数列表的 Promise
Promise.all([myFetch(), myFetch()])

// 返回最先运行完的 myFetch 的结果,只有一个
Promise.race([myFetch(), myFetch()])

上面两个方法还不够灵活,无法实现并发数的控制,下面提供两个思路。

1. Promise.all

Promise.all能并发执行参数列表中的 Promise,那么我们可以控制参数列表中 Promise 的数量,实现类似并发数控制的功能。

const tasks = Array.from({length: 10}).map((_, idx) => {   
  return () => new Promise(resolve => {
      setTimeout(() => {
      	console.log(idx, 'done')
      	resolve(idx)
      }, idx * 100)
  })
})

async function Scheduler(promises, limit) { 
  const results = []
  
  for(let i=0; i<=promises.length; i+=limit) {
  	const tasks = []
  	// 切片 Promise.all 的参数
    for(const promise of promises.slice(i, i + limit)) {
      tasks.push(promise())
	}
	// 用 await 以确保后面的任务不会在执行当前任务时执行
    const result = await Promise.all(tasks)
    results.push(...result)
  }
  
  return results
}

这种方法的基本思路就是通过 for 循环切片出指定数量的任务,然后 await Promise.all(tasks) 阻塞当前函数的运行。使用这种方法有如下注意事项:

  1. promises 的参数必须是一个包含返回 promise 对象的函数
    这是因为如果直接传入 Promise 对象,这个 Promise 会立即执行,无法达到并发控制的效果。
  2. 后一批并发的任务必须等待前一批任务全部完成才能执行,这是因为 Promise.all 的功能限制。从这一点来讲当任务数量多于并发数时,会产生时间上的浪费。

手写并发控制类

这种实现方法下,我们保存一个任务列表,当运行中的任务的数量小于并发数时,从任务列表中取出任务并执行。

class Scheduler {
	
	// 并发运行 tasks,最大并发量由 limit 指定
	// 每次运行前得先 new 一个实例出来,应该 tasks set 没有重置
	runningTasks = new Set()
	fufilledTasks = new Set()
	
	run(tasks, limit) {
		for(const task of tasks) {
			if(
				this.runningTasks.size<= limit && 
				!this.fufilledTasks.has(task)
			){
				this.runningTasks.add(task)
				this.fufilledTasks.add(task)
				
				task().then(() => {
					this.runningTasks.delete(task)
					this.run(tasks, limit)
				})
			}
		}
	}
}

这种情况下我们先保存没运行的协程,在那些被运行的协程运行后的 .then 方法中,运行没运行的协程,达到并发控制的目的。
可以优化的点:

  1. 提供返回值列表
  2. 调用 run 方法之前无需实例化类
  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
实现 `promise.map` 并限制 promise并发数,可以使用 `Promise.all` 结合 `Promise.race` 来实现。 首先,将要处理的数据数组分割成多个小数组,每个小数组的长度等于并发数。然后,使用 `Array.map` 方法对每个小数组创建一个 Promise,将 Promise 存放在一个新的数组中。 接下来,使用 `Promise.all` 来等待所有 Promise 完成,并返回一个新的 Promise。在这个新的 Promise 中,使用 `Promise.race` 来限制并发数,每次只执行指定数量的 Promise。当其中一个 Promise 完成时,会触发 `Promise.race` 返回的 Promise,然后继续执行下一个 Promise,以此类推。 下面是一个示例的实现代码: ```javascript function promiseMap(array, concurrency, mapper) { const chunks = []; for (let i = 0; i < array.length; i += concurrency) { chunks.push(array.slice(i, i + concurrency)); } const promises = chunks.map(chunk => { return Promise.all(chunk.map(mapper)); }); return Promise.all(promises).then(results => { return [].concat(...results); }); } ``` 使用示例: ```javascript const data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]; function asyncTask(item) { return new Promise(resolve => { setTimeout(() => { console.log(`Processing item: ${item}`); resolve(item * 2); }, 1000); }); } promiseMap(data, 3, asyncTask).then(results => { console.log('All tasks completed:', results); }); ``` 在上述示例中,`promiseMap` 函数将 `data` 数组中的每个元素传递给 `asyncTask` 函数进行处理,并限制了并发数为 3。输出结果会按照并发数限制的顺序进行处理,每个任务间隔 1 秒执行。 这样,你就可以实现一个带有并发限制的 `promise.map` 函数。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值