背景
提到异常,我们会想到:抛出异常、异常恢复、资源清理、吞掉异常、重新抛出异常、替换异常、包装异常。本文想谈谈 “包装异常”,主要针对这个问题:何时应该 “包装异常”?
“包装异常” 的技术形式
包装异常是替换异常的特殊形式,具体的技术形式如下:
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#,确实是一部好书,关于异常作为契约部分的想法,和作者产生了很大的共鸣,书中对异常处理的讲解非常细致,推荐大家读一读这本书。