Dart库充满了返回Future
或Stream
对象的函数。这些函数是异步的:它们在设置一个可能耗时的操作(如I / O )后返回,而无需等待该操作完成。
async
和await
关键字支持异步编程,允许你编写看起来类似于同步代码的异步代码。
处理Futures
当你需要一个完整Futures的结果时,你有两个选择:
- 使用
async
和await
- 使用Futures的API
使用async
和await
的代码是异步的,但是看起来很像同步代码。例如,以下代码使用await
来等待异步函数的结果:
await lookUpVersion();
使用await
,代码必须是在一个*异步函数(async function)里-标记为async
的异步函数:
Future checkVersion() async {
var version = await lookUpVersion();
// 用version变量做点什么
}
注意:虽然异步函数可能会执行耗时的操作,但它不会等待这些操作。相反,异步函数只在遇到它的第一个
await
表达式之前执行。然后,它返回一个Future的对象,只有在await
表达式完成后,才恢复执行。
使用try
,catch
和finally
,最后在使用await
的代码去处理错误和清理后续逻辑
try {
version = await lookUpVersion();
} catch (e) {
// 无法查找到version做出反应
}
你可以在异步函数中多次使用await
。例如,下面的代码等待函数结果三次:
var entrypoint = await findEntrypoint();
var exitCode = await runExecutable(entrypoint, args);
await flushThenExit(exitCode);
在await 表达式
中,表达式
的值通常是一个Future;如果不是,则该值将在Future中自动包装。这个Future的对象表示承诺返回一个对象。await expression
的值就是那返回的对象。await表达式使程序运行停止,直到对象可用!
如果当你使用await
的时候,遇到到一个编译时错误。此时请确保await
是否处于异步函数中。例如,在app的main()
函数中使用await
,则main()
主主体必须标志为async
即异步
Future main() async {
checkVersion();
print('In main: version is ${await lookUpVersion()}');
}
声明异步函数
当一个函数主体被async
修饰符标志,那么这个函数就是一个异步函数
将async
关键字添加到函数中会使它返回Future。例如,看下这个同步函数,它返回一个字符串:
String lookUpVersion() => '1.0.0';
请注意,函数的主体不需要使用Future的API。如果需要,dart会创建Future的对象。
如果你的函数没有任何有用的值,请将其返回类型设为Future<void>
。
处理流
当你需要从流中获取值时,有两个选项:
- 使用
async
和异步循环(await for
) - 使用流API
注意:在使用
await for
之前,请确保它能使代码更加清晰,并且你确实希望等待流的所有结果。例如,你通常不应该为UI事件侦听器使用await for
,因为UI框架发送无尽的事件流。
异步for循环具有以下形式:
await for (varOrType identifier in expression) {
//每次流发出值时执行
}
表达式
的值必须具有类型流。执行过程如下:
- 等到流发出一个值。
- 执行for循环的主体,将变量设置为该发出的值。
- 重复第1步和第2步,直至流关闭。
若要停止侦听流,可以使用break
或return
语句,该语句脱离for循环并取消订阅流。
如果在实现异步for循环时遇到编译时错误,请确保await for
在异步函数中。例如,要在应用程序的main()
函数中使用异步for循环,main()
的主体必须标记为async(异步):
Future main() async {
// ...
await for (var request in requestServer) {
handleRequest(request);
}
// ...
}
有关异步编程的更多信息,一般来说,请参见库教程的dart:async
。另请参见文章Dart Language Asynchrony Support: Phase 1和 Dart Language Asynchrony Support: Phase 2,以及 Dart language specification
。
PS:本文
整理
自官方文档,若有发现问题请致邮 caoyanglee92@gmail.com