考虑以下代码。
import java.io.IOException;
import java.sql.SQLException;
public class ReassignmentICatch {
public void couldThrowAnException() throws SQLException, IOException {}
public void rethrow() throws SQLException, IOException {
try {
couldThrowAnException();
System.out.println("Did not throw");
} catch (Exception e) { //Line-1
e = new IOException(); //Line-2
throw e; //Line-3
}
}
public static void main(String[] args) {
ReassignmentICatch rc = new ReassignmentICatch();
try {
rc.rethrow();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
据我所知,java7 中的 catch(Exception e)
并不像 java6 那样捕获所有异常,而是仅捕获方法声明中提到的那些异常 couldThrowAnException
。此外,不允许将新异常重新分配给 catch 参数 (e),这就是我们在 Line-3
处出现编译错误的原因
java6 中也不允许这种重新分配吗?我认为应该允许在 java6 中重新分配,所以我在 eclipse 中更改了 java-compiler 的属性并将合规级别设置为 1.6 以查看此代码是否编译。
我无法预测的行为是:
1.Eclipse 报同样的错误
Unhandled exception type Exception
适用于 Java 6 7 和 8。
2.当我尝试使用命令行时 javac -target 1.6 -source 1.6 ReassignmentICatch.java
正在获取
warning: [options] bootstrap class path not set in conjunction with -source 1.6 ReassignmentICatch.java:18: error: unreported exception Exception; must be caugh t or declared to be thrown
throw e; ^ 1 error 1 warning
并且使用 java7 和 java8 我能够编译并成功运行它并输出 Did not throw
。为什么 IDE 和命令行没有给出相同的结果。
有人可以建议我在这里缺少什么吗?
谢谢。
最佳答案
我不确定您所说的 Java 6 和 Java 7 在捕获异常方面的不同之处是什么意思,因为我不知道类似的事情,您可以发布引用吗?
方法声明中提到的异常是throws clause
异常。这些是方法本身不捕获但希望它的调用者捕获的异常。
分配给 catch 变量时不应该出现错误。但是,您遇到的错误是因为您抛出了异常类型,而它未在 throws 子句中声明。
考虑下面的代码
e = new IOException(); //Line-2
throw e; //Line-3
即使您已经分配了 e
IOException
实例,e
的类型仍然是 Exception
。哪些未被捕获且未在 throws 子句中声明,因此报告为错误。
有两种解决方法。您可以在 throws 子句中声明 Exception 或在抛出之前强制转换 e
。
public void rethrow() throws SQLException, IOException
//becomes
public void rethrow() throws Exception
或
throw e;
//becomes
throw (IOExceptoin)e;
//or
throw new IOException();
编辑*
link 中给出的 Java 6 和 Java 7 之间的区别是 Java 7 能够在编译时推断异常类型。即使它被分配给一个父类(super class)类型,Java 7 也能够推断出它的原始类型并且不会提示。这使我们不必将异常对象包装到 RuntimeException
对象中。
Eclipse 和命令行输出不同的最可能原因是您没有针对不同的 Java 版本修改 Eclipse 中的编译器设置。在 Eclipse 中更改项目的 Java 版本时,您需要更改以下内容
- 构建路径中的 Java 系统库
- 编译器源代码合规性(首选项>Java>编译器>编译器合规性级别)
完成这两项更改后,您应该能够看到正确的行为。
关于java - 将异常重新分配给 catch 参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34460773/