List.forEach()中的异步/等待

ael*_*yeb 12 asynchronous dart

我正在编写某种机器人(命令行应用程序),当我使用"forEach"方法时,我遇到异步执行问题.这是我正在尝试做的简化代码:

main() async {
  print("main start");
  await asyncOne();
  print("main end");
}

asyncOne() async {
  print("asyncOne start");
  [1, 2, 3].forEach(await (num) async {
    await asyncTwo(num);
  });
  print("asyncOne end");
}

asyncTwo(num) async
{
  print("asyncTwo #${num}");
}
Run Code Online (Sandbox Code Playgroud)

这是输出:

main start
asyncOne start
asyncOne end
main end
asyncTwo #1
asyncTwo #2
asyncTwo #3
Run Code Online (Sandbox Code Playgroud)

我想要得到的是:

main start
asyncOne start
asyncTwo #1
asyncTwo #2
asyncTwo #3
asyncOne end
main end
Run Code Online (Sandbox Code Playgroud)

如果有人知道我做错了什么,我会很感激.

Ste*_*ane 48

您需要使用Future.forEach.

main() async {
  print("main start");
  await asyncOne();
  print("main end");
}

asyncOne() async {
  print("asyncOne start");
  await Future.forEach([1, 2, 3], (num) async {
    await asyncTwo(num);
  });
  print("asyncOne end");
}

asyncTwo(num) async
{
  print("asyncTwo #${num}");
}
Run Code Online (Sandbox Code Playgroud)

  • 这应该是批准的答案。 (5认同)
  • 对于阅读本文的任何人,请注意,不幸的是,这仅适用于可迭代。 (2认同)
  • 你刚刚节省了我在互联网上调试和搜索的时间。 (2认同)
  • 请注意,虽然这是最好的方法,但编译器可能不知道块中“num”的类型。因此,您可能必须调用类似 `await Future.forEach<int>(someIntList, (item) async {` (2认同)

stw*_*ton 22

我不认为用这种forEach方法实现你想要的东西.但是它可以使用for循环.例;

asyncOne() async {
  print("asyncOne start");
  for (num number in [1, 2, 3])
    await asyncTwo(number);
  print("asyncOne end");
}
Run Code Online (Sandbox Code Playgroud)


小智 13

我知道这是一个老问题,但我会在这里留下一个新答案,希望这对将来的人有所帮助。

您可以通过执行以下操作将 forEach用于您想要实现的目标:

  asyncOne() async {
  print("asyncOne start");
  await Future.forEach([1, 2, 3],(num) async {
    await asyncTwo(num);
  });
  print("asyncOne end");
}
Run Code Online (Sandbox Code Playgroud)

  • 这实际上是@Stephane 上面发布的内容的一个子集。 (2认同)

lrn*_*lrn 6

你无法使用forEach它,因为它实际上并没有查看其回调的返回值.如果它们是期货,它们就会丢失,而不是等待.

您可以像Steven Upton建议的那样进行循环,也可以使用,Future.wait如果您希望操作同时运行,而不是一个接一个地运行:

asyncOne() async {
  print("asyncOne start");
  await Future.wait([1, 2, 3].map(asyncTwo));
  print("asyncOne end");
}
Run Code Online (Sandbox Code Playgroud)