Async/ждет в List.forEach()

Я пишу какой-то бот (приложение командной строки), и у меня возникают проблемы с выполнением async, когда я использую метод 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}");
}

И вот вывод:

main start
asyncOne start
asyncOne end
main end
asyncTwo #1
asyncTwo #2
asyncTwo #3

То, что я пытаюсь получить, это:

main start
asyncOne start
asyncTwo #1
asyncTwo #2
asyncTwo #3
asyncOne end
main end

Если кто-то знает, что я делаю неправильно, я был бы признателен.

Ответ 1

Я не думаю, что можно добиться того, чего вы хотите с помощью метода forEach. Однако он будет работать с контуром for. Пример

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

Ответ 2

Вам нужно использовать 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}");
}

Ответ 3

Вы не можете использовать forEach для этого, потому что на самом деле он не смотрит на возвращаемые значения своих обратных вызовов. Если они являются фьючерсами, они просто будут потеряны и не ожидаются.

Вы можете либо сделать цикл, как предложил Стивен Аптон, либо использовать Future.wait, если вы хотите, чтобы операции выполнялись одновременно, а не один за другим:

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

Ответ 4

Я знаю, что это старый вопрос, но я оставлю здесь новый ответ, надеясь, что это поможет кому-то в будущем.

Вы можете использовать forEach для того, чего вы пытаетесь достичь, выполнив что-то вроде этого:

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