简体   繁体   English

JavaScript / TypeScript - 承诺循环不等待

[英]JavaScript/TypeScript - loop of promises doesn't wait

I have a loop of 100 promises. 我有一个100承诺的循环。 I want to process 10 of those simultaneously, then print a statement, and continue with the next 10 in parallel. 我想同时处理其中的10个,然后打印一个语句,然后并行继续下一个10。 However, it does not wait for every 10 promises to finish their execution. 但是,它不会等待每10个承诺完成执行。

const promises = [];
for (let i = 1; i <= 100; i ++) {

   const blockPromise = this.web3.eth.getBlock(i).then((block: any) => {
        winston.info("Processing block " + i);
   }).catch((err: Error) => {
        winston.error(err);
   });

   promises.push(blockPromise);

   if (i % 10 === 0) {
       Promise.all(promises).then(() => {
            winston.info("+++ Processed " + 10 + " blocks");
       }).catch((err: Error) => {
            winston.error(err);
       });
   }

My expectation is something like: 我的期望是这样的:

Processing block 1
Processing block 3
Processing block 7
Processing block 2
Processing block 4
Processing block 5
Processing block 6
Processing block 9
Processing block 10
Processing block 8
+++ Processed 10 blocks
Processing block 12
...

However, it is chaotically mixed up. 然而,它混乱起来。 What am I doing wrong? 我究竟做错了什么? Any help appreciated. 任何帮助赞赏。

Another solution is to use array reduce to chain your promises: 另一个解决方案是使用array reduce来链接你的承诺:

 var groups = Array.from(new Array(10),(val,index) => 10*index); // => [0, 10, 20, 30, 40, 50, 60, 70, 80, 90] groups.reduce((m, o) => { return m.then(() => { var group = Array.from(new Array(10), (val, index) => o + index + 1); // [1, 2, 3, 4, 5, 6, 7, 8, 9, 10], [11, 12, 13, 14, 15, 16, 17, 18, 19, 20], ... var promises = group.map(i => this.web3.eth.getBlock(i).then(() => winston.info("Processing block " + i)) ); return Promise.all(promises).then(() => winston.info("+++ Processed " + 10 + " blocks") ); }); }, Promise.resolve(true)); 

It just gives you the expected result: 它只是给你预期的结果:

在此输入图像描述

Since you are trying to process 10 promises each time, you could await after each batch of 10 or would that defeat the purpose? 既然你每次尝试处理10个承诺,你可以在每批10个之后await或者是否会失败?

var promises = []; // You are mutating it so better not be a const
var blockPromise = null;
for (let i = 1; i <= 100; i ++) {

 blockPromise = this.web3.eth.getBlock(i).then((block: any) => {
      winston.info("Processing block " + i);
 }).catch((err: Error) => {
      winston.error(err);
 });

 promises.push(blockPromise);

 if (i % 10 === 0) {
     await Promise.all(promises);
     promises = [] // clear the processed promises as well
     winston.info("+++ Processed " + 10 + " blocks");
 }
}

Promises inside for loops mess up everything. 内部循环的承诺搞砸了一切。 Either you use async and await, or you need to reduce the whole thing to one promise chain: 您可以使用async和await,或者您需要将整个事务减少到一个promise链:

let promise = Promise.resolve(), promises = [];
for( let i = 0; i <= 100; i++){
  promises.push(this.web3.eth.getBlock(i).then((block: any) => {
         winston.info("Processing block " + i);
     }).catch((err: Error) => {
         winston.error(err);
  }));

  if (i % 10 === 0){
   promise = Promise.all(promises.concat(promise))
      .then(_=>winston.info("+++ Processed " + 10 + " blocks");  
   promises = [];           
  }
}

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM