熟悉了c#的异步方法后,各种是不是就会有一种一发不可收拾的情况
对于我们熟悉的 async/await 方法,我们知道await可以执行一个异步方法,并且在方法结束后“继续”当前方法之后的代码。 这个对于执行耗时操作,且不阻塞当前线程的情形非常有效,例如
foo()
{
await A_async();
B();
}
那么假设下如果方法A_async会抛出异常,那怎么办。
很简单用trycatch,可以很好处理这些异常
foo()
{
try
{
await A_async();
}
catch(Exception e)
{
Console.WriteLine(e);
}
B();
}
但是如果期望是在外部catch异常会如何?
foo()
{
await A_async();
B();
}
foo2()
{
try
{
foo();
}
catch(Exception e)
{
Console.WriteLine(e);
}
}
答案是无法catch。
虽然A_async()的异常是在foo方法内部抛出,但是在遇到await关键字时,foo2的线程就开始“分叉”成2个线程执行。所以可以认为,foo2的线程已经完成的他的try代码块
那么第二个问题,如果现在是方法B会抛出异常,而A_async不会,那么下面两种写法那种会被catch?
foo()
{
await A_async().ConfigureAwaiter(True);
B();
}
foo2()
{
try
{
foo();
}
catch(Exception e)
{
Console.WriteLine(e);
}
}
foo()
{
await A_async().ConfigureAwaiter(False);
B();
}
foo2()
{
try
{
foo();
}
catch(Exception e)
{
Console.WriteLine(e);
}
}
铛铛,都不会被catch。不管B是否在同步上下文执行,外层的catch代码都不会进行等待。
但是以下情况就会被catch咯
foo()
{
B();
await A_async().ConfigureAwaiter(False);
}
foo2()
{
try
{
foo();
}
catch(Exception e)
{
Console.WriteLine(e);
}
}
因为代码在遇到await 关键字之前,实际还在trycatch语句块之中。
本文会经常更新,请阅读原文: https://xinyuehtx.github.io/post/%E8%AF%B7%E4%B8%8D%E8%A6%81%E4%BE%9D%E8%B5%96%E4%BA%8E%E5%BC%82%E6%AD%A5%E4%BB%A3%E7%A0%81%E7%9A%84%E8%B0%83%E7%94%A8%E6%96%B9catch%E5%BC%82%E5%B8%B8.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名黄腾霄(包含链接: https://xinyuehtx.github.io ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 。