经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » JS/JS库/框架 » Omi » 查看文章
js Promise并发控制数量的方法
来源:jb51  时间:2021/8/26 17:21:10  对本文有异议

问题

要求写一个方法控制 Promise 并发数量,如下:

  1. promiseConcurrencyLimit(limit, array, iteratorFn)

limit 是同一时间执行的 promise 数量,array 是参数数组,iteratorFn 每个 promise 中执行的异步操作。

背景

开发中需要在多个promise处理完成后执行后置逻辑,通常使用Promise.all:

  1. Primise.all([p1, p2, p3]).then((res) => ...)

但是有个问题是,因为 promise 创建后会立即执行,也就是说传入到 promise.all 中的多个 promise 实例,在其创建的时候就已经开始执行了,如果这些实例中执行的异步操作都是 http 请求,那么就会在瞬间发出 n 个 http 请求,这样显然是不合理的;更合理的方式是:对 Promise.all 中异步操作的执行数量加以限制,同一时间只允许有 limit 个异步操作同时执行。

思路 & 实现

在背景中提到,promise 在创建后就会立即执行,所以控制并发的核心在于控制 promise 实例的生成。最开始只生成 limit 个 promise 实例,然后等待这些 promise 状态变更,只要其中某一个 promise 实例的状态发生变更,就立即再创建一个 promise 实例...如此循环,直到所有的 promise 都被创建并执行。

npm 上有很多库实现了此功能,个人觉得 tiny-async-pool 这个库比较好,因为它直接使用了原生的 Promise 实现了此功能,而其他库大多重新实现了 promise。其核心代码如下:

  1. async function asyncPool(poolLimit, array, iteratorFn) {
  2. const ret = []; // 用于存放所有的promise实例
  3. const executing = []; // 用于存放目前正在执行的promise
  4. for (const item of array) {
  5. const p = Promise.resolve(iteratorFn(item)); // 防止回调函数返回的不是promise,使用Promise.resolve进行包裹
  6. ret.push(p);
  7. if (poolLimit <= array.length) {
  8. // then回调中,当这个promise状态变为fulfilled后,将其从正在执行的promise列表executing中删除
  9. const e = p.then(() => executing.splice(executing.indexOf(e), 1));
  10. executing.push(e);
  11. if (executing.length >= poolLimit) {
  12. // 一旦正在执行的promise列表数量等于限制数,就使用Promise.race等待某一个promise状态发生变更,
  13. // 状态变更后,就会执行上面then的回调,将该promise从executing中删除,
  14. // 然后再进入到下一次for循环,生成新的promise进行补充
  15. await Promise.race(executing);
  16. }
  17. }
  18. }
  19. return Promise.all(ret);
  20. }

测试代码如下:

  1. const timeout = (i) => {
  2. console.log('开始', i);
  3. return new Promise((resolve) => setTimeout(() => {
  4. resolve(i);
  5. console.log('结束', i);
  6. }, i));
  7. };
  8.  
  9. (async () => {
  10. const res = await asyncPool(2, [1000, 5000, 3000, 2000], timeout);
  11. console.log(res);
  12. })();
  13.  

代码的核心思路为:

  • 先初始化 limit 个 promise 实例,将它们放到 executing 数组中
  • 使用 Promise.race 等待这 limit 个 promise 实例的执行结果
  • 一旦某一个 promise 的状态发生变更,就将其从 executing 中删除,然后再执行循环生成新的 promise,放入executing 中
  • 重复2、3两个步骤,直到所有的 promise 都被执行完
  • 最后使用 Promise.all 返回所有 promise 实例的执行结果

到此这篇关于js Promise并发控制数量的方法的文章就介绍到这了,更多相关js Promise并发控制内容请搜索w3xue以前的文章或继续浏览下面的相关文章希望大家以后多多支持w3xue!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号