请不要依赖于异步代码的调用方catch异常

本文探讨C#异步编程中关于异常处理的问题。当使用async/await时,异常处理机制有所不同。如果异步方法A_async抛出异常,外部catch无法捕获。即使在try-catch块中,await后的方法执行会在不同线程,导致异常不会被捕获。同样的,如果方法B抛出异常,两种常见写法也无法由外部catch处理。只有当await位于try块内时,异常才会被正确捕获。建议阅读原文以获取更详细信息并避免误解。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

熟悉了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/ ,以避免陈旧错误知识的误导,同时有更好的阅读体验。

知识共享许可协议 本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名黄腾霄(包含链接: https://xinyuehtx.github.io/ ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值