InvocationTargetException 存在的意义是什么?

文档里说这个异常是 `Method` 和 `Constructor` 抛出来的,大部分使用 invoke 的时候都是包装一下然后重新抛出它的 cause,这样设计是有什么原因吗?为何不能够直接抛出原来的异常?
关注者
18
被浏览
1824
因为“原来的异常”无法直接以一种统一而又明确的方式表达出来,所以使用InvocationTargetException来将原来的异常包装起来,通过多加一层间接层的方式来提供统一的访问途径。

JavaDoc一上来就说了:
InvocationTargetException is a checked exception that wraps an exception thrown by an invoked method or constructor.

Java方法可以静态声明它可能会抛出一组固定的异常类型。而反射API里,Method.invoke() 与 Constructor.newInstance() 这些方法有“双重身份”——它们既代表要调用指定的目标方法,自身也是一个方法;目标方法可能会抛出异常,而它们自身在调用目标方法前也可能会抛出一些异常(例如IllegalArgumentException)。
它们要调用的目标方法可能抛出任意Throwable类的派生类的异常,但它们自身却不能根据要调用的目标而“动态”改变自己声明要抛出的异常类型,而只能静态声明一组可能抛出的异常类型。
声明抛出Throwable或Exception的话,这就太宽泛,难以准确反映异常的原因和意图;但不声明成这么宽泛的异常类型的话又无法完整覆盖所有可能由目标方法抛出的异常。

那怎么办?简单,新增一个check exception类型,InvocationTargetException,将原本由目标方法抛出的异常包装起来,这样就可以给Method.invoke() / Constructor.newInstance() 的调用者一个统一的接口,既明确了“这个异常是由目标方法抛出的,不是由我自己抛出的”的意图,又能完整覆盖目标方法所能抛出的所有异常类型(InvocationTargetException.getTargetException() / getCause() 的类型是Throwable)。