javascript - typescript - while 循环 for Promise

标签 javascript loops typescript promise

我有一个大数组,它被分成更小的 block 。然后我对每个 block 进行异步调用:

  myFunc(arrObjs: any[], chunkSize: number): Promise<any> {
    let result = Promise.resolve(); // To start the chain

    while(arrObjs.length) {
      // Extracting array chunk
      let chunk = arrObjs.splice(0, chunkSize);
      // Building the chain
      result = result.then(() => {
        return someAsynFunc(
          chunk
        );
      });
    }
    return result;
  }

以上是我的解决方案。我需要按顺序运行异步函数(系列 promise ),因此我构建了一系列 promise 。

有没有更好的方法来做到这一点?

最佳答案

是的,您可以使用reduce模式:

function myFunc(arrObjs: any[], chunkSize: number): Promise<any> {
    const chunks : any[] = [];
    while (arrObjs.length) {
        chunks.push(arrObjs.splice(0, chunkSize));
    }

    return chunks.reduce((p : Promise<any>, chunk : any[]) => {
        return p.then(() => someAsyncFunc(chunk));
    }, Promise.resolve());
  }

(我认为我的类型注释是正确的;如果没有,希望您可以阅读它......)

请注意,我保留了您原来的行为,实际上是从 arrObjs 数组中删除数据,但通常我不提倡像这样更改从调用者传递的对象。

这是一个例子:

function myFunc(arrObjs, chunkSize) {
  const chunks = [];
  while (arrObjs.length) {
    chunks.push(arrObjs.splice(0, chunkSize));
  }

  return chunks.reduce((p, chunk) => {
    return p.then(() => someAsyncFunc(chunk));
  }, Promise.resolve());
}

function someAsyncFunc(chunk) {
  console.log("Start handling", JSON.stringify(chunk));
  return new Promise(resolve => {
    setTimeout(() => {
      console.log("Done handling", JSON.stringify(chunk));
      resolve();
    }, 800);
  });
}

const array = [0, 1, 2, 4, 5, 6, 7, 8, 9];
myFunc(array, 3).then(() => {
  console.log("Done");
});

<小时/>

另请注意,上面的代码在处理它们之前提前从数组中删除所有 block 。如果您想在处理它们时执行此操作(这会与使用该数组的其他任何内容产生串扰,这可能会添加/删除/等),您可以做同样的事情,只是没有减少:

// Cross-talky version
function myFunc(arrObjs : any[], chunkSize : number) : Promise<any> {
  return new Promise<any>(function processChunk(resolve : function) {
    const chunk : any[] = arrObjs.splice(0, chunkSize);
    if (chunk.length == 0) {
      resolve();
    } else {
      someAsyncFunc(chunk).then(() => processChunk(resolve));
    }
  });
}

// Cross-talky version
function myFunc(arrObjs, chunkSize) {
  return new Promise(function processChunk(resolve) {
    const chunk = arrObjs.splice(0, chunkSize);
    if (chunk.length == 0) {
      resolve();
    } else {
      someAsyncFunc(chunk).then(() => processChunk(resolve));
    }
  });
}

function someAsyncFunc(chunk) {
  console.log("Start handling", JSON.stringify(chunk));
  return new Promise(resolve => {
    setTimeout(() => {
      console.log("Done handling", JSON.stringify(chunk));
      resolve();
    }, 800);
  });
}

const array = [0, 1, 2, 4, 5, 6, 7, 8, 9];
myFunc(array, 3).then(() => {
  console.log("Done");
});
setTimeout(() => {
  array.splice(2, 0, 'cross', 'talk');
}, 1000);

关于javascript - typescript - while 循环 for Promise,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40994658/

相关文章:

javascript - 通过表单将用户发送到付款 URL

javascript - Table .append() 同时发生

javascript - 如何遍历所有 <p> 元素?

c++ - 循环比较(优化)

mysql - ASP.NET如何在div中显示sqldatasource行?

javascript - 如何在 Angular2 中申请日期?

javascript - 在 iFrame 中重新加载同一站点的路由

typescript - RxjS shareReplay : how to reset its value?

c# - 使用 http.get 以 Angular 与 asp.net Web api 下载 .xlsx 文件时出错

reactjs - material-ui 对话框内的表单未在 Enter 上提交