编写高质量代码改善C#程序的157个建议——建议60:重新引发异常时使用Inner Exception

建议60:重新引发异常时使用Inner Exception

当捕获了某个异常,将其包装或重新引发异常的时候,如果其中包含了Inner Exception,则有助于程序员分析内部信息,方便代码调试。

以一个分布式系统为例,在进行远程通信的时候,可能会发生的情况肯能会有:

1)网卡被禁用或者网线断开,此时会抛出SocketException,消息问:“由于目标机器积极拒绝,无法连接。”

2)网络正常,但是要链接的目标主机没有端口没有处在监听状态,此时会抛出SocketException,消息为:“由于连接方在一段时间后没有正确答复或连接的主机没有反应,连接尝试失败。”

3)链接超时,此时需要通过代码实现关闭链接,并抛出SocketException,消息为:“链接超过约定的时长。”

发生以上三种情况的任何一种情况,在返回给最终用户的时候,我们都需要将异常信息包装成为“网络连接失败,请稍后再试”。

所以,一个分布式系统的业务处理方法,看起来应该是这样的:

            try
            {
                SaveUser(user);
            }
            catch (SocketException ex)
            {
                throw new CommunicationFailureException("网络连接失败,请稍后再试", ex);
            }

在提示这条信息的时候,我们可能需要将原始异常记录到日志里,以供开发者分析具体原因(如果这种情况频繁出现,就可能是一个Bug)。在记录日志时,非常有必要记录此异常出现的内部异常或是堆栈信息。

throw new CommunicationFailureException("网络连接失败,请稍后再试", ex);就是将异常包装成为一个CommunicationFailureException,并将SocketException作为InnerException(即ex)向上传递。

如果不打算使用InnerException,但是要返回一下额外的信息,可以使用Exception的Data属性。如下:

            try
            {
                SaveUser(user);
            }
            catch (SocketException ex)
            {
                ex.Data.Add("SocketInfo","网络连接失败,请稍后再试");
                throw ex;
            }

在上层进行捕获的时候,可以通过键值来得到异常信息:

            catch (SocketException ex)
            {
                Console.WriteLine(ex.Data["SocketInfo"]);
            }

转自:《编写高质量代码改善C#程序的157个建议》陆敏技