Flutter异步代码中的执行顺序错误

时间:2019-08-05 12:23:59

标签: asynchronous flutter dart future

我在Flutter应用中遇到了期货问题。

void saveCats() async {
  var cats = await convertToCats(_rawData);
  await DatabaseProvider.db.addCats(cats);
}

Future<List<Cat>> convertToCats(CatList catData) async {
  var cats = <Cat>[];
  await catData.forEach(key, value) async {
    var pos = await calculatePos();
    print('This should come first');
    cats.add(Cat(name: key, pos: pos);
  }
}

Future<int> calculatePos() async {
  return await DatabaseProvider.db.calculatePos();
}

database.dart:

Future<void> addCats(List<Cat> cats) async {
  print('This should come second');
  // Do adding stuff here
}

Future<int> calculatePos() async {
  // Some code to calculate the position
  return pos;
}

在上面的代码中,点击按钮时将调用saveCats函数。此功能将一些原始数据转换为Cat模型列表,并将其添加到数据库中。作为此转换过程的一部分,它将计算列表中猫的pos。我的问题是,我希望在我的两个print语句中,forEach循环中的一个应该在addCats数据库函数中的一个之前。但是相反,它们以相反的顺序出现。我要去哪里错了?

2 个答案:

答案 0 :(得分:2)

您不能在List.forEach()或Map.forEach()中async/await,因为它们两个都返回void。 要么使用

await Future.forEach([1, 2, 3], (num) async {
    await asyncMethod(num);
  });

或类似的东西; https://api.flutter.dev/flutter/dart-async/Future/forEach.html

答案 1 :(得分:2)

forEach通常无法满足您的期望,因为提供的函数作为闭包运行。

当您要遍历列表对每个元素做一些事情以使用for(或诸如map等功能性更强的方法之一)时,这是更自然的选择。

目前尚不清楚CatList是什么类型,所以这是近似值,但是您会想要更多类似的东西:

Future<List<Cat>> convertToCats(CatList catData) async {
  var cats = <Cat>[];
  for (var i = 0; i < catData.length; i++) {
    var key = catData[i].key;
    var pos = await calculatePos();
    print('This should come first');
    cats.add(Cat(name: key, pos: pos));
  }
  return cats;
}

Future<List<Cat>> convertToCats(CatList catData) async {
  return catData.map(...some mapping function...).toList();
}