1. 程式人生 > >利用反射呼叫方法丟擲的異常如何被捕獲?

利用反射呼叫方法丟擲的異常如何被捕獲?

我們通常在java開發中採用自定義異常,在業務中遇到非系統錯誤時丟擲自定義異常,並在上層進行捕獲,就能知道業務的具體出錯資訊。這種方法很常用,但是如果採用反射去自動呼叫某個方法時,卻不能捕獲到使用者自定義的異常。下面用程式碼來說明:
/**
* 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
    } 
   }
}

通常我們是這樣捕獲異常,並輸出資訊的。
期望的結果是:
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,知道這個問題的原因就能很好的處理異常情況了。