Dart async*、yield* 和异常处理
Dart async*, yield* and exception handling
如果我有嵌套的 async* 流,异常似乎是不可捕获的,这是违反直觉的。
一个例子:
void main() {
getString().listen(print);
}
Stream<String> getString() async* {
try {
yield* asyncStarError();
yield await asyncError();
} catch (err) {
yield 'Crash';
}
}
Stream<String> asyncStarError() async* {
throw Exception('A Stream error happened');
}
Future<String> asyncError() async {
throw Exception('A Future error happened');
}
这输出:
Uncaught Error: Exception: A Stream error happened
Crash
所以asyncStarError
抛出的异常没有被捕获,而Future却如期被捕获。谁能解释一下为什么?
您可以在 dartpad 中观察行为:https://dartpad.dartlang.org
yield*
转发它产生的流中的所有事件,包括错误。
因此,asyncStarError()
生成一个带有错误事件的流,yield*
将该错误转发给 getString
返回的流,然后 getString.listen(print);
不添加处理程序,因此错误未被捕获。
(我猜你是在浏览器中 运行 因为那个未捕获的错误不会使程序崩溃。)
在那之后,yield await asyncError()
再也没有产生任何东西。 await asyncError()
本身在到达 yield
之前抛出,然后该错误被 catch
捕获,产生 crash
.
如果你想捕捉一个流的错误,你需要真正地看事件,而不是仅仅使用yield*
一味地转发它们,包括错误事件。
例如:
await for (var _ in asyncStarError()) {
// Wohoo, event!
}
会使来自 asyncStarError
流的错误成为循环中的错误。然后它会被捕获,你会打印“崩溃”。
TL;DR: yield*
操作转发错误事件,它不会在本地引发它们。
不同之处在于异步生成器 (async*) 异常无法通过用 try/catch 包围来捕获。
相反,您应该使用回调 handleError 来实现您正在寻找的行为。
更进一步,您可能有兴趣使用 runZonedGuarded。
因为 yield * 像@lrn 说的那样转发它们,所以你可以在 main 中捕获它们。或者无论你在哪里消费它们。
void main() {
getString().listen(print).onError((e) => print('error caught: $e'));
}
Stream<String> getString() async* {
try {
yield* asyncStarError();
yield await asyncError();
} catch (err) {
yield 'Crash';
}
}
Stream<String> asyncStarError() async* {
throw Exception('A Stream error happened');
}
Future<String> asyncError() async {
throw Exception('A Future error happened');
}
这会打印:
error caught: Exception: A Stream error happened
Crash
如果我有嵌套的 async* 流,异常似乎是不可捕获的,这是违反直觉的。
一个例子:
void main() {
getString().listen(print);
}
Stream<String> getString() async* {
try {
yield* asyncStarError();
yield await asyncError();
} catch (err) {
yield 'Crash';
}
}
Stream<String> asyncStarError() async* {
throw Exception('A Stream error happened');
}
Future<String> asyncError() async {
throw Exception('A Future error happened');
}
这输出:
Uncaught Error: Exception: A Stream error happened
Crash
所以asyncStarError
抛出的异常没有被捕获,而Future却如期被捕获。谁能解释一下为什么?
您可以在 dartpad 中观察行为:https://dartpad.dartlang.org
yield*
转发它产生的流中的所有事件,包括错误。
因此,asyncStarError()
生成一个带有错误事件的流,yield*
将该错误转发给 getString
返回的流,然后 getString.listen(print);
不添加处理程序,因此错误未被捕获。
(我猜你是在浏览器中 运行 因为那个未捕获的错误不会使程序崩溃。)
在那之后,yield await asyncError()
再也没有产生任何东西。 await asyncError()
本身在到达 yield
之前抛出,然后该错误被 catch
捕获,产生 crash
.
如果你想捕捉一个流的错误,你需要真正地看事件,而不是仅仅使用yield*
一味地转发它们,包括错误事件。
例如:
await for (var _ in asyncStarError()) {
// Wohoo, event!
}
会使来自 asyncStarError
流的错误成为循环中的错误。然后它会被捕获,你会打印“崩溃”。
TL;DR: yield*
操作转发错误事件,它不会在本地引发它们。
不同之处在于异步生成器 (async*) 异常无法通过用 try/catch 包围来捕获。
相反,您应该使用回调 handleError 来实现您正在寻找的行为。
更进一步,您可能有兴趣使用 runZonedGuarded。
因为 yield * 像@lrn 说的那样转发它们,所以你可以在 main 中捕获它们。或者无论你在哪里消费它们。
void main() {
getString().listen(print).onError((e) => print('error caught: $e'));
}
Stream<String> getString() async* {
try {
yield* asyncStarError();
yield await asyncError();
} catch (err) {
yield 'Crash';
}
}
Stream<String> asyncStarError() async* {
throw Exception('A Stream error happened');
}
Future<String> asyncError() async {
throw Exception('A Future error happened');
}
这会打印:
error caught: Exception: A Stream error happened
Crash