突破Java異常處理規則
突破Java異常處理規則[@more@] 問題: 我在我的應用程式中呼叫了外部方法並且想捕獲它可能丟擲的異常。我能捕獲java.lang.Exception嗎?
答案: 透過一個給定的方法去處理所有執行時和檢測異常對於預防外部錯誤是不充分的。
你可以去讀目前 JavaWorld文章 – “Java Tip 134: When Catching Exception, Don’t Cast Your Net Too Wide”。這篇文章警告了捕獲java.lang.Exception和java.lang.Throable是不好的。捕獲你能指定的異常對於程式碼的可維護性是十分重要的。然而這個規則依賴於特殊的環境。如果你不打算你的程式崩潰並且保留你的資料結構的安全異常,那麼你必須捕獲被丟擲的真正的異常。
舉個例子,想象你有一個載入了這個介面的伺服器應用:
public interface IFoo
{
/**
* This method can't throw any checked exceptions...or can it?
*/
void bar ();
} // End of interface
對於給出引數的理由是讓我們通知你這樣的服務在什麼地方,並且不同的IFoo實現能夠從外部資源載入上。你寫如下程式碼:
try
{
IFoo foo = ... // get an IFoo implementation
foo.bar ();
}
catch (RuntimeException ioe)
{
// Handle 'ioe' ...
}
catch (Error e)
{
// Handle or re-throw 'e' ...
}
並且你在這個裡處理了所有可能的異常。你不需要在這裡加上任何捕獲java.io.IOException的異常,因為IFoo實現沒有從IFoo.bar()中丟擲它,對嗎?(事實上,如果你加上了捕獲java.io.IOException異常塊,編譯器可能會把它作為不可到達的異常而丟棄)
錯誤。在我寫的EvilFoo類中bar()方法證明了將丟擲你傳遞給類構造器的任何異常:
public void bar ()
{
EvilThrow.throwThrowable (m_throwthis);
}
執行Main方法:
public class Main
{
public static void main (final String[] args)
{
// This try/catch block appears to intercept all exceptions that
// IFoo.bar() can throw; however, this is not true
try
{
IFoo foo = new EvilFoo (new java.io.IOException ("SURPRISE!"));
foo.bar ();
}
catch (RuntimeException ioe)
{
// Ignore ioe
}
catch (Error e)
{
// Ignore e
}
}
} // End of class
你將看到從bar()方法丟擲的java.io.IOException異常例項並且沒有任何捕獲塊:
>java -cp classes Main
Exception in thread "main" java.io.IOException: SURPRISE!
at Main.main(Main.java:23)
在這裡發生了什麼?
主要的觀察是通常針對檢測異常的Java規則僅僅在編譯的時候被執行。在執行的時候,一個JVM不能保證被一個方法丟擲的異常是否和在這個方法中宣告的丟擲異常相匹配。因為呼叫方法的職責是捕獲和處理所有從呼叫方法丟擲的異常。任何沒有被呼叫方法宣告的異常將不予理睬並且拒絕呼叫棧。
答案: 透過一個給定的方法去處理所有執行時和檢測異常對於預防外部錯誤是不充分的。
你可以去讀目前 JavaWorld文章 – “Java Tip 134: When Catching Exception, Don’t Cast Your Net Too Wide”。這篇文章警告了捕獲java.lang.Exception和java.lang.Throable是不好的。捕獲你能指定的異常對於程式碼的可維護性是十分重要的。然而這個規則依賴於特殊的環境。如果你不打算你的程式崩潰並且保留你的資料結構的安全異常,那麼你必須捕獲被丟擲的真正的異常。
舉個例子,想象你有一個載入了這個介面的伺服器應用:
public interface IFoo
{
/**
* This method can't throw any checked exceptions...or can it?
*/
void bar ();
} // End of interface
對於給出引數的理由是讓我們通知你這樣的服務在什麼地方,並且不同的IFoo實現能夠從外部資源載入上。你寫如下程式碼:
try
{
IFoo foo = ... // get an IFoo implementation
foo.bar ();
}
catch (RuntimeException ioe)
{
// Handle 'ioe' ...
}
catch (Error e)
{
// Handle or re-throw 'e' ...
}
並且你在這個裡處理了所有可能的異常。你不需要在這裡加上任何捕獲java.io.IOException的異常,因為IFoo實現沒有從IFoo.bar()中丟擲它,對嗎?(事實上,如果你加上了捕獲java.io.IOException異常塊,編譯器可能會把它作為不可到達的異常而丟棄)
錯誤。在我寫的EvilFoo類中bar()方法證明了將丟擲你傳遞給類構造器的任何異常:
public void bar ()
{
EvilThrow.throwThrowable (m_throwthis);
}
執行Main方法:
public class Main
{
public static void main (final String[] args)
{
// This try/catch block appears to intercept all exceptions that
// IFoo.bar() can throw; however, this is not true
try
{
IFoo foo = new EvilFoo (new java.io.IOException ("SURPRISE!"));
foo.bar ();
}
catch (RuntimeException ioe)
{
// Ignore ioe
}
catch (Error e)
{
// Ignore e
}
}
} // End of class
你將看到從bar()方法丟擲的java.io.IOException異常例項並且沒有任何捕獲塊:
>java -cp classes Main
Exception in thread "main" java.io.IOException: SURPRISE!
at Main.main(Main.java:23)
在這裡發生了什麼?
主要的觀察是通常針對檢測異常的Java規則僅僅在編譯的時候被執行。在執行的時候,一個JVM不能保證被一個方法丟擲的異常是否和在這個方法中宣告的丟擲異常相匹配。因為呼叫方法的職責是捕獲和處理所有從呼叫方法丟擲的異常。任何沒有被呼叫方法宣告的異常將不予理睬並且拒絕呼叫棧。
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/10901326/viewspace-965638/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- Java異常處理12條軍規Java
- JAVA 異常處理Java
- Java 異常處理Java
- JAVA異常處理Java
- JAVA_異常處理Java
- Java 異常表與異常處理原理Java
- Java異常處理(非常全面)Java
- Java異常處理機制Java
- 重學Java - 異常處理Java
- java異常處理筆記Java筆記
- 異常篇——異常處理
- Java 的異常處理機制Java
- Java之異常處理try{}catch(){}Java
- Java進階02 異常處理Java
- JAVA學習之異常處理Java
- java異常的處理機制Java
- Java 異常處理:使用和思考Java
- 異常處理
- Java入門教程十一(異常處理)Java
- 甩鍋(throws)_ java異常(Exception)處理JavaException
- Java 中的異常處理機制Java
- [轉載] Java異常處理習題Java
- JSP 異常處理如何處理?JS
- 異常-throws的方式處理異常
- React 異常處理React
- JS異常處理JS
- oracle異常處理Oracle
- Python——異常處理Python
- Python異常處理Python
- ThinkPHP 異常處理PHP
- JavaScript 異常處理JavaScript
- 異常的處理
- golang - 異常處理Golang
- 異常處理2
- 異常處理1
- Abp 異常處理
- 08、異常處理
- SpringMVC異常處理SpringMVC
- 異常處理機制(二)之異常處理與捕獲