一文搞懂 Promise 并发控制:批量执行 vs 最大并发数,实用场景全解析!

在前端开发中,批量处理异步请求是常见需求,比如批量上传图片、爬取数据、导入大批量信息等。但如果不加控制,可能会遇到接口限流、浏览器卡顿、甚至服务端拒绝请求等问题。今天带你用最简单的方式,掌握两种常用的 Promise 并发控制技巧,并结合实际开发场景,分析常见坑点和解决办法!

场景一:批量执行(分批并发)

应用场景举例:

  • 批量上传图片时,后端只允许每次最多5个请求,否则会返回"请求过多"错误。
  • 需要分批导入数据,保证每批都能被后端稳定处理。

常见问题:

  • 一次性全部发起请求,容易被限流或接口报错。
  • 没有分批,用户体验差,失败率高。

解决思路:将任务分成一批一批,每批最多 N 个,上一批全部完成后再执行下一批。

实现代码如下:

js 复制代码
// 并发分批执行,每批limit个,上一批全部完成后再执行下一批
function batchRequest(tasks, limit) {
  const results = new Array(tasks.length);
  let index = 0;

  const runBatch = () => {
    if (index >= tasks.length) return Promise.resolve();
    const batch = tasks.slice(index, index + limit).map((task, i) => {
      const realIndex = index + i;
      return Promise.resolve(task())
        .then(value => ({ status: 'fulfilled', value }))
        .catch(reason => ({ status: 'rejected', reason }))
        .then(result => {
          results[realIndex] = result;
        });
    });
    index += limit;
    return Promise.all(batch).then(() => {
      if (index < tasks.length) {
        console.log('本批次执行完毕,将要执行下一个批次');
      }
      return runBatch();
    });
  }

  return runBatch().then(() => results);
}

// 模拟异步任务
const mockTask = (id, duration) => {
  return () =>
    new Promise(resolve => {
      console.log(`[排队] 任务${id} 进入队列,需要 ${duration}ms`);
      setTimeout(() => resolve(`任务${id}结果`), duration);
    });
};

// 创建测试任务
const tasks = [
  mockTask(1, 2000),
  mockTask(2, 1500),
  mockTask(3, 800),
  mockTask(4, 1200),
  mockTask(5, 3000),
  mockTask(6, 500),
  mockTask(7, 1800),
  mockTask(8, 600),
  mockTask(9, 900),
  mockTask(10, 700),
];

// 用法示例
batchRequest(tasks, 3).then(results => {
  console.log('====== 所有任务完成 ======');
  console.log('最终结果:', results);
});

运行结果:

场景二:最大并发数控制(随时补位)

应用场景举例:

  • 爬虫抓取数据时,既要快又不能让服务器崩溃
  • 前端批量请求接口,既想高效又要防止被限流

常见问题

  • 没有限流,接口容易被封
  • 并发数太低,效率又太低

解决思路:始终保持 N 个任务在执行,有空位就立刻补新任务,直到全部完成。

实现代码如下:

js 复制代码
// 最大并发数控制,随时补位
function controlRequest(tasks, limit) {
  return new Promise(resolve => {
    const results = new Array(tasks.length);
    let currentIndex = 0;
    let completeCount = 0;
    let currentRunCount = 0;

    const runTask = () => {
      while (currentIndex < tasks.length && currentRunCount < limit) {
        const taskIndex = currentIndex++;
        currentRunCount++;
        const currentTask = tasks[taskIndex];
        console.log(`[启动] 任务${taskIndex + 1} 开始 (并发数:${currentRunCount})`);
        Promise.resolve(currentTask())
          .then(value => { results[taskIndex] = { status: 'fulfilled', value }; })
          .catch(reason => { results[taskIndex] = { status: 'rejected', reason }; })
          .finally(() => {
            completeCount++;
            currentRunCount--;
            if (completeCount === tasks.length) {
              resolve(results);
            } else {
              runTask();
            }
          });
      }
    }
    runTask();
  });
}

// 模拟异步任务
const mockTask = (id, duration) => {
  return () =>
    new Promise(resolve => {
      console.log(`[排队] 任务${id} 进入队列,需要 ${duration}ms`);
      setTimeout(() => resolve(`任务${id}结果`), duration);
    });
};

// 创建测试任务
const tasks = [
  mockTask(1, 2000),
  mockTask(2, 1500),
  mockTask(3, 800),
  mockTask(4, 1200),
  mockTask(5, 3000),
  mockTask(6, 500),
  mockTask(7, 1800),
  mockTask(8, 600),
  mockTask(9, 900),
  mockTask(10, 700),
];
// 用法示例
controlRequest(tasks, 3).then(results => {
  console.log('====== 所有任务完成 ======');
  console.log('最终结果:', results);
});

运行结果:

欢迎在评论区留言交流,如果觉得有用,记得点赞+关注

相关推荐
拉里呱唧7 分钟前
一个像在使用PPT的在线 HTML 编辑器:HeyHTML
javascript·交互·html5
We་ct14 分钟前
LeetCode 97. 交错字符串:动态规划详解
前端·算法·leetcode·typescript·动态规划
Chengbei1123 分钟前
轻量化 Web 安全日志分析神器 星川智盾日志威胁检测、地理溯源、MITRE ATT&CK 映射,支持 Windows/macOS/Linux
前端·人工智能·安全·web安全·macos·系统安全·安全架构
风流 少年24 分钟前
Python Web框架:FastAPI
前端·python·fastapi
GISer_Jing32 分钟前
AI时代面试新常态——从“会用工具”到“深挖原理”的跨越
前端·人工智能·ai编程
IT_陈寒42 分钟前
React的useEffect把我坑惨了,这些闭包陷阱真要命
前端·人工智能·后端
前端之虎陈随易1 小时前
有生之年系列,Nodejs进程管理pm2 v7.0发布
前端·typescript·npm·node.js
ayqy贾杰1 小时前
Cursor SDK发布!开发者可直接搬走其内核
前端·vue.js·面试
changshuaihua0011 小时前
扣子开发指南
javascript·人工智能
椰猫子1 小时前
SpringMVC(SpringMVC简介、请求与响应(请求映射路径、请求参数、日期类型参数传递、响应json数据))
java·前端·数据库