你可以去读目前 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实现能够从外部资源加载上。你写如下代码:
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不能保证被一个方法抛出的异常是否和在这个方法中声明的抛出异常相匹配。因为调用方法的职责是捕捉和处理所有从调用方法抛出的异常。任何没有被调用方法声明的异常将不予理睬并且拒绝调用栈。