Skip to content

分享流行的前端产品

技术|源码|设计|参考

公众号二维码

扫码关注艾小逗公众号

第一时间传递给你

首页/javascript/本文内容

js控制并发请求的最优解

思路:维护一个运行池,一个等待队列,出一个进一个,控制运行池的大小

html
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>并发请求示例</title>
</head>
<body>
    <script>
        // 模拟请求
        const requestFn = () => {
            return new Promise((resolve) => {
                setTimeout(() => {
                    resolve(true)
                }, 50);
            })
        }
        const urlList = new Array(100).fill(requestFn)

        const poll = new Set();
        const waitQueue = [];
        const request = url => {
            return new Promise((res, rej) => {
                const isFull = poll.size >= 10;
                const fn = function fn(){
                    // const request = fetch(url);
                    const request = url();
                    request.finally(() => {
                        poll.delete(fn);
                        const next = waitQueue.shift();
                        next && poll.add(next);
                        setTimeout(() => next?.())
                    })
                    request.then(res)
                    request.catch(rej)
                    return request;
                }
                if (isFull) {
                    waitQueue.push(fn)
                } else {
                    poll.add(fn);
                    fn();
                }
            })
        }


        for(let [index, url] of urlList.entries()) {
            request(url).then(() => console.log("====", index))
        }

    </script>
</body>
</html>

控制调用频率

工作中还有一种情况,非并发,是控制调用api的频率 需求: 数据库中存的2000多条地图经纬度数据有的不准确,所以想把所有的数据都重新获取重新刷一下,控制下调用百度地图api 的频率,每100毫秒调用一次,然后提交给接口进行保存。 实现方式: 使用setTimeout实现,延迟100毫秒 * index 执行【如果大家有其他实现方式请提供下,谢谢】

js
  import waitData from './waitData.json';  // 所有待执行的数据,2000多条数据
  let waitSaveData: any = [];
  let successData: any = [];
  let waitSaveErrInfo: any = [];
  
  // 保存失败的数据文件,不含失败信息,方便再次尝试执行
  function saveErrToFile() {
    const blob = new Blob([JSON.stringify({ result: { records: waitSaveData } })], { type: 'text/plain' });
    const link = document.createElement('a');
    link.href = URL.createObjectURL(blob);
    link.download = 'data.json';
    link.click();
    URL.revokeObjectURL(link.href);
  }
  // 保存失败的数据文件,含失败信息
  function saveErrInfoToFile() {
    const blob = new Blob([JSON.stringify(waitSaveErrInfo)], { type: 'text/plain' });
    const link = document.createElement('a');
    link.href = URL.createObjectURL(blob);
    link.download = 'err.json';
    link.click();
    URL.revokeObjectURL(link.href);
  }
 // 保存成功的数据文件
  function saveSuccessToFile() {
    const blob = new Blob([JSON.stringify(successData)], { type: 'text/plain' });
    const link = document.createElement('a');
    link.href = URL.createObjectURL(blob);
    link.download = 'err.json';
    link.click();
    URL.revokeObjectURL(link.href);
  }


// 刷新数据方法
const refreshInfo = () => {
    waitSaveData = [];
    successData = [];
    waitSaveErrInfo = [];
    console.log('✨开始刷新数据✨', waitData.result.records);
    const allDataLength = waitData.result.records.length;
    waitData.result.records.forEach((item, index) => {
      var bdary = new BMap.Boundary(); // 百度地图api,通过城市获取行政区域的经纬度数据
      setTimeout(() => { // 定时器,延迟100毫秒 * index 执行
        bdary.get(item.putGoodsPlace, function (rs) {
          var count = rs.boundaries.length; //行政区域的点有多少个
          if (count === 0) { // 获取失败,保存失败数据
            console.info(item.putGoodsPlace + '未能获取当前输入行政区域');
            waitSaveData.push(item);
            waitSaveErrInfo.push({ ...item, err: '未能获取当前输入行政区域' });
            return;
          }
          var boundLngLat = JSON.stringify(rs.boundaries);
          var lngLat = '';
          for (var i = 0; i < count; i++) {
            lngLat += rs.boundaries[i] + ';';
          }
          lngLat = lngLat.substring(0, lngLat.length - 1);
          // 要提交接口的数据
          const data = {
            lngLat,
            boundLngLat,
            type: '1',
            figureType: '1',
            targetPlaceCode: item.targetPlaceCode,
            targetPlaceName: item.targetPlaceNameCp,
            enclosureName: item.enclosureName,
            putGoodsPlace: item.putGoodsPlace,
            warehouseName: item.warehouseName,
            longLatCode: item.longLatCode,
          };
          saveData(data)
            .then(() => {
              successData.push(item);
            })
            .catch((err) => {
              console.log('✨file:index.vue:211✨✨', err.message);
              waitSaveData.push(item);
              waitSaveErrInfo.push({ data: { ...data, lngLat: '请查看请求', boundLngLat: '请查看请求' }, err: err.message });
            });
            // 打印索引,方便查看执行进度
          if (index % 50 === 0) {
            console.log('当前索引', index);
          }
          if (index === allDataLength - 1) {
            console.log('✨执行完毕✨', index);
          }
        });
      }, 100 * index);
    });
  };

waitData.json 文件格式如下:

{
    success: true,
    message: '操作成功',
    code: 200,
    result: {
      records: [
        {
          longLatCode: 'BWL000000002058863',
          enclosureName: '广东省韶关市南雄市',
          putGoodsPlace: '广东省韶关市南雄市',
          type: '1',
          isUser: '1',
          warehouseName: '',
          radius: '',
          figureType: '1',
          targetPlaceCode: '440000,440200,440200',
          targetPlaceNameCp: '广东省,韶关市,南雄市',
        },
        {
          longLatCode: 'BWL000000002058511',
          enclosureName: '辽宁省沈阳市法库县',
          putGoodsPlace: '辽宁省沈阳市法库县',
          type: '1',
          isUser: '1',
          warehouseName: '',
          radius: '',
          figureType: '1',
          targetPlaceCode: '210000,210100,210100',
          targetPlaceNameCp: '辽宁省,沈阳市,法库县',
        },
      ],
    },
  };