Q: 我在我的应用程序中调用了外部方法并且想捕获它可能抛出的异常。我能捕获java.lang.Exception吗?
A: 通过一个给定的方法去处理所有运行时和检测异常对于预防外部错误是不充分的。
你可以去读目前 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不能保证被一个方法抛出的异常是否和在这个方法中声明的抛出异常相匹配。因为调用方法的职责是捕获和处理所有从调用方法抛出的异常。任何没有被调用方法声明的异常将不予理睬并且拒绝调用栈。
如果正常行为是编译器执行,那么我怎么创建EvilFoo的?至少有两个方法可以去创建抛出没有声明的异常的Java方法:
Thread.stop(Throwable)在一些时候不被赞成使用,但是它仍然被使用并且传递一个Throwable给被调用的Thread。
分别编译:你能在编译EvilFoo时候不去编译真正声明bar()方法抛出检测异常的IFoo临时版本。
我用后一种选择:我编译开始定义的EvilThrow类:
public abstract class EvilThrow { public static void throwThrowable (Throwable throwable) throws Throwable { throw throwable; } } |
public void dispatch() { if (catchExceptions) { try { runnable.run(); } catch (Exception e) { exception = e; } } else { runnable.run(); } if (notifier != null) { synchronized (notifier) { notifier.notifyAll(); } } } |
public static void invokeAndWait(Runnable runnable) throws InterruptedException, InvocationTargetException { class AWTInvocationLock {} Object lock = new AWTInvocationLock(); InvocationEvent event =new InvocationEvent(Toolkit.getDefaultToolkit(), runnable, lock, true); synchronized (lock) { Toolkit.getEventQueue().postEvent(event); lock.wait(); } Exception eventException = event.getException(); if (eventException != null) { throw new InvocationTargetException(eventException); } } |
public void run () { bar (); } |
SwingUtilities.invokeAndWait (new EvilFoo (new Throwable ("SURPRISE!"))); |