这可能看起来像一个编程101问题,我以为我知道答案,但现在发现自己需要再次检查。在下面这段代码中,第一个捕获块中抛出的异常会被下面的通用异常捕获块捕获吗?

try {
  // Do something
} catch(IOException e) {
  throw new ApplicationException("Problem connecting to server");
} catch(Exception e) {
  // Will the ApplicationException be caught here?
}

我一直认为答案是否定的,但现在我有一些奇怪的行为可能是由这引起的。大多数语言的答案可能都是一样的,但我用的是Java。


当前回答

它不会被第二个catch块捕获。每个异常仅在try块内捕获。你可以尝试嵌套(并不是说这是一个好主意):

try {
    doSomething();
} catch (IOException) {
   try {
       doSomething();
   } catch (IOException e) {
       throw new ApplicationException("Failed twice at doSomething" +
       e.toString());
   }          
} catch (Exception e) {
}

其他回答

Java语言规范在14.19.1节中说:

If execution of the try block completes abruptly because of a throw of a value V, then there is a choice: If the run-time type of V is assignable to the Parameter of any catch clause of the try statement, then the first (leftmost) such catch clause is selected. The value V is assigned to the parameter of the selected catch clause, and the Block of that catch clause is executed. If that block completes normally, then the try statement completes normally; if that block completes abruptly for any reason, then the try statement completes abruptly for the same reason.

参考: http://java.sun.com/docs/books/jls/second_edition/html/statements.doc.html#24134

换句话说,第一个可以处理异常的封闭捕获可以,如果从该捕获抛出异常,则该异常不在原始try的任何其他捕获的范围内,因此它们不会尝试处理它。

需要知道的一件相关且令人困惑的事情是,在try-[catch]-finally结构中,finally块可能会抛出异常,如果是这样,try或catch块抛出的任何异常都会丢失。第一次看到的时候可能会感到困惑。

不,因为新的throw不是直接在try块中。

旧帖子,但“e”变量必须是唯一的:

try {
  // Do something
} catch(IOException ioE) {
  throw new ApplicationException("Problem connecting to server");
} catch(Exception e) {
  // Will the ApplicationException be caught here?
}

不——正如Chris Jester-Young所说,它将被抛到等级中的下一个尝试接球。

如上所述…… 我想补充的是,如果您无法看到正在发生的事情,如果您无法在调试器中重现问题,那么您可以在重新抛出新异常之前添加跟踪(最糟糕的情况是使用老式的system .out.println,否则使用像log4j这样的日志系统)。