Java 语言规范 11.2.2对最终异常和非最终异常进行了区分:
一个 throw 语句(第 14.18 节),其抛出的表达式具有静态类型 E,并且不是 final 或有效的 final 异常参数,可以抛出 E 或抛出的表达式可以抛出的任何异常类。
[...]
抛出的表达式是 catch 子句 C 的最终或有效最终异常参数的 throw 语句可以抛出异常类 E iff:
- E是声明C的try语句的try块可以抛出的异常类;和
- E 是与 C 的任何可捕获异常类兼容的赋值;和
- E 与在同一 try 语句中声明在 C 左侧的 catch 子句的任何可捕获异常类的赋值都不兼容。
有趣的是,JLS 14.20还说:
在 uni-catch 子句中,未声明为 final(隐式或显式)的异常参数如果从未作为赋值运算符的左侧操作数出现在其范围内,则被认为是有效的 final。
换句话说,如果你不重新分配e
你的 catch 语句(如e = new SomeOtherException();
),它会被隐式声明为 final。
所以我只能得出结论,它没有任何区别,除非在 catch 块中修改了异常,我能想出的唯一例子是:
public void method1() throws IOException {
try {
throw new IOException();
} catch (Exception e) { // e is not modified in catch => implicitly final
throw e; //compiles OK
}
}
//it works because method1 is semantically equivalent to method2:
public void method2() throws IOException {
try {
throw new IOException();
} catch (final Exception e) {
throw e;
}
}
public void method3() throws IOException {
try {
throw new IOException("1");
} catch (Exception e) {
e = new IOException("2"); //e modified: not implicitly final any more
throw e; //does not compile
}
}