利用反射调用方法抛出的异常如何被捕获?
我们通常在java开发中采用自定义异常,在业务中遇到非系统错误时抛出自定义异常,并在上层进行捕获,就能知道业务的具体出错信息。这种方法很常用,但是如果采用反射去自动调用某个方法时,却不能捕获到用户自定义的异常。下面用代码来说明:
/**
* WebException为自定义异常
*/
通常我们是这样捕获异常,并输出信息的。
期望的结果是:
n < 0
但是实际结果为
null
其实java认为用反射来调用方法时,jvm不能在编译期间确定方法的throws 类型,所以方法可能抛出的异常jvm也不能动态确定其类型,而统一抛出InvocationTargetException(注意特殊情况请参考)。那么我们怎么修改以上代码能正确地获取到业务抛出的异常信息呢?
修改标记1处的异常块:
其实关键的点就是利用反射技术调用方法时总是抛出InvocationTargetException,知道这个问题的原因就能很好的处理异常情况了。
/**
* WebException为自定义异常
*/
class Foo{
public void fun(int n) throws WebException {
if( n >= 0)
return;
else
throw new WebException("n < 0");
}
}
public class Test {
public static void main(String[] args){
try{
int n = -1;
Object foo = Class.forName("com.libra.web.servlet.Foo").newInstance();
Class c = foo.getClass();
Method m = c.getMethod("fun", new Class []{int.class});
m.invoke(foo, new Object[]{new Integer (n)});
}catch(Exception ex){
System.out.println(ex.getMessage()); // 标记1
}
}
}
public void fun(int n) throws WebException {
if( n >= 0)
return;
else
throw new WebException("n < 0");
}
}
public class Test {
public static void main(String[] args){
try{
int n = -1;
Object foo = Class.forName("com.libra.web.servlet.Foo").newInstance();
Class c = foo.getClass();
Method m = c.getMethod("fun", new Class []{int.class});
m.invoke(foo, new Object[]{new Integer (n)});
}catch(Exception ex){
System.out.println(ex.getMessage()); // 标记1
}
}
}
通常我们是这样捕获异常,并输出信息的。
期望的结果是:
n < 0
但是实际结果为
null
其实java认为用反射来调用方法时,jvm不能在编译期间确定方法的throws 类型,所以方法可能抛出的异常jvm也不能动态确定其类型,而统一抛出InvocationTargetException(注意特殊情况请参考)。那么我们怎么修改以上代码能正确地获取到业务抛出的异常信息呢?
修改标记1处的异常块:
}catch(Exception ex){
InvocationTargetException targetEx = (InvocationTargetException)ex;
Throwable t = targetEx .getTargetException();
System.out.println( (WebException) targetEx.getMessage());
}
InvocationTargetException targetEx = (InvocationTargetException)ex;
Throwable t = targetEx .getTargetException();
System.out.println( (WebException) targetEx.getMessage());
}
其实关键的点就是利用反射技术调用方法时总是抛出InvocationTargetException,知道这个问题的原因就能很好的处理异常情况了。