接着上次没说完的异常来看。我们说,当代码抛出一个异常时,就会终止方法中剩余代码的处理,并退出这个方法的执行。 但是有一种情况,如果方法获得了一些本地资源,并且只有这个方法自己知道,并且这些资源在退出方法之前必须被回收,那么就存在一个资源回收的问题。
对于这个问题,Java提出了一个finally子句来解决问题:
不管是否有异常被捕获,finally子句中的代码都被执行。我们来看个例子:
InputStream in = new FileInputStream(...)
try
{
//1
code that might throw exceptions
//2
}
catch(IOException e){
//3
show error message
//4
}
finally{
//5
in.close();
}
//6
无论在try语句块中是否遇到异常,finally子句中的in.close()语句都会被执行。
这里,我们强烈建议解耦合try/catch和try/finally语句块。 如下所示:
InputStream in = ...;
try
{
try
{
code that might throw exceptions;
}
finlly
{
in.close();
}
}
catch(IOException e)
{
show error message
}
这是值得肯定的设计方式。
但是如果在try语句块中抛出的并不是IOException异常,比如调用close时,抛出了一个close异常,该怎么办呢?
在java7之前,我们是这样解决的:
InpuntStream in = ...;
Exception ex = null;
try
{
try{
code that might throw exception
}
catch(Exception e){
ex = e;
throw e;
}
}finally
{
try
{
in.close();
}
catch(Exception e){
if(ex == null)throw e;
}
}
但是在java7中,推出了带资源的try语句。
假设资源属于一个实现了AutoCloseable接口的类,Java SE 7为这种代码模式提供了一种快捷方式:
AutoCloseable接口有一个方法:void close() throws Exception
try块退出时,会自动调用res.close()。下面给出一个例子,这里读取一个文件中所有单词:
try(Scanner in = new Scanner(new FileInputStream("/usr/share/dict/words")),"UTF-8")
{
while(in.hasNext())
System.out.println(in.next());
}
这个块正常退出时,或者存在一个异常时,都会调用in.close()方法,就好像使用了finally块一样。
上一段代码的try语句还可以指定多个资源。例如
try(Scanner in = new Scanner(new FileInputStream("/usr/share/dict/words"),"UTF-8");
PrintWriter out = new PrintWriter("out.txt"))
{
...
}
同样的,不管这个块如何退出,in和out都会关闭。