Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 589|回复: 0

.NET:何时应该 “包装异常”?

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-9-3 15:14:34 | 显示全部楼层 |阅读模式

    背景

    提到异常,我们会想到:抛出异常、异常恢复、资源清理、吞掉异常、重新抛出异常、替换异常、包装异常。本文想谈谈 “包装异常”,主要针对这个问题:何时应该 “包装异常”?

    “包装异常” 的技术形式

    包装异常是替换异常的特殊形式,具体的技术形式如下:

    1             try
    2             {
    3                 // do something
    4             }
    5             catch (SomeException ex)
    6             {
    7                 throw new WrapperException("New Message", ex);
    8             }

    注意:WrapperException 需要将 ex 作为 InnerException,这样才不至于丢失 StackTrace,WrapperException.StackTrace 和 ex.StackTrace 共同构成了完整的 StackTrace。

    让例子帮助我们得出答案

    有这样一种场景:我希望为各种 ORM 框架提供一种抽象,这可以让应用开发人员自由的在不同的 ORM 实现之间做出选择。

    第一个版本的实现

    实现伪代码

     1     interface IRepository<TEntity>
     2     {
     3         void Update(TEntity entity);
     4     }
     5 
     6     class EntityFrameworkRepository<TEntity> : IRepository<TEntity>
     7     {
     8         public void Update(TEntity entity)
     9         {
    10             throw new EntityFrameworkConcurrentException(); // 可能会抛出这样的异常,这里的代码不是十分准确。
    11         }
    12     }
    13 
    14     class NHibernateRepository<TEntity> : IRepository<TEntity>
    15     {
    16         public void Update(TEntity entity)
    17         {
    18             throw new NHibernateConcurrentException(); // 可能会抛出这样的异常,这里的代码不是十分准确。
    19         }
    20     }

    有什么问题?

    处理并发异常是应用层开发人员的一个非常重要的职责,他们或者选择自动重试、或者选择让用户重试、甚至允许并发带来的不一致性,如果使用了上面的接口问题就大了,应用中该拦截哪种并发异常呢?EntityFrameworkConcurrentException?NHibernateConcurrentException?这样的接口和实现无论如何都达不到:OCP 和 LSP。

    将异常作为契约的一部分

    实现伪代码

     1     interface IRepository<TEntity>
     2     {
     3         void Update(TEntity entity);
     4     }
     5 
     6     class ConcurrentException : Exception
     7     {
     8     }
     9 
    10     class EntityFrameworkRepository<TEntity> : IRepository<TEntity>
    11     {
    12         public void Update(TEntity entity)
    13         {
    14             try
    15             {
    16             }
    17             catch (EntityFrameworkConcurrentException ex)
    18             {
    19                 throw new ConcurrentException(ex);
    20             }
    21         }
    22     }
    23 
    24     class NHibernateRepository<TEntity> : IRepository<TEntity>
    25     {
    26         public void Update(TEntity entity)
    27         {
    28             try
    29             {
    30             }
    31             catch (NHibernateConcurrentException ex)
    32             {
    33                 throw new ConcurrentException(ex);
    34             }
    35         }
    36     }

    有什么问题?

    目前来说还觉得不错,如果 C# 编译器或 CLR 能支持异常契约就好了,Java 虽然支持,但是对于调用者来说又不太友好。

    这里给出答案

    当异常是契约的一部分时,才需要包装异常。

    可能还会有其它答案,等我再思考思考,朋友们也可以给出一些想法。

    微软的一个反例

     MethodBae.Invoke

    1         //   System.Reflection.TargetInvocationException:
    2         //     调用的方法或构造函数引发异常。
    3         //
    4         //   System.MethodAccessException:
    5         //     调用方没有调用此构造函数的权限。
    6         //
    7         //   System.InvalidOperationException:
    8         //     声明此方法的类型是开放式泛型类型。 即,System.Type.ContainsGenericParameters 属性为声明类型返回 true。
    9         public abstract object Invoke(object obj, BindingFlags invokeAttr, Binder binder, object[] parameters, CultureInfo culture);

    当方法内部抛出异常时,Invoke 会将内部异常给包装起来,这明显不是我们期望的行为,后来微软的 dynamic 调用 和 CreateDelegate 之后使用 Delegate 调用 都修复了这个问题。

    备注

    最近在读第四版的 clr via c#,确实是一部好书,关于异常作为契约部分的想法,和作者产生了很大的共鸣,书中对异常处理的讲解非常细致,推荐大家读一读这本书。

     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-21 15:43 , Processed in 0.073450 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表