用finally做什么
在没有“垃圾收集”以及“自动挪用粉碎器”机制的一种语言中(注释⑤),finally显得出格重要,因为措施员可用它包管内存的正确释放——无论在try块内部产生了什么状况。但Java提供了垃圾收集机制,所以内存的释放险些绝对不会成为问题。别的,它也没有构建器可供挪用。既然如此,Java里何时才会用到finally呢?
⑤:“粉碎器”(Destructor)是“构建器”(Constructor)的反义词。它代表一个非凡的函数,一旦某个工具失去用处,凡是就会挪用它。我们必定知道在那边以及何时挪用粉碎器。C++提供了自动的粉碎器挪用机制,但Delphi的Object Pascal版本1及2却不具备这一本领(在这种语言中,粉碎器的寄义与用法都产生了变革)。
除将内存设回原始状态以外,若要配置另一些对象,finally就是必须的。譬喻,我们有时需要打开一个文件可能成立一个网络毗连,可能在屏幕上画一些对象,甚至配置外部世界的一个开关,等等。如下例所示:
//: OnOffSwitch.java
// Why use finally?
class Switch {
boolean state = false;
boolean read() { return state; }
void on() { state = true; }
void off() { state = false; }
}
public class OnOffSwitch {
static Switch sw = new Switch();
public static void main(String[] args) {
try {
sw.on();
// Code that can throw exceptions...
sw.off();
} catch(NullPointerException e) {
System.out.println("NullPointerException");
sw.off();
} catch(IllegalArgumentException e) {
System.out.println("IOException");
sw.off();
}
}
} ///:~
这里的方针是担保main()完成时开关处于封锁状态,所以将sw.off()置于try块以及每个违例节制器的末端。但发生的一个违例有大概不是在这里捕捉的,这便会错过sw.off()。然而,操作finally,我们可以未来自try块的封锁代码只置于一个处所:
//: WithFinally.java
// Finally Guarantees cleanup
class Switch2 {
boolean state = false;
boolean read() { return state; }
void on() { state = true; }
void off() { state = false; }
}
public class WithFinally {
static Switch2 sw = new Switch2();
public static void main(String[] args) {
try {
sw.on();
// Code that can throw exceptions...
} catch(NullPointerException e) {
System.out.println("NullPointerException");
} catch(IllegalArgumentException e) {
System.out.println("IOException");
} finally {
sw.off();
}
}
} ///:~
在这儿,sw.off()已移至一个处所。无论产生什么工作,都必定会运行它。
纵然违例不在当前的catch从句集里捕捉,finally城市在违例节制机制转到更高级别搜索一个节制器之前得以执行。如下所示:
//: AlwaysFinally.java
// Finally is always executed
class Ex extends Exception {}
public class AlwaysFinally {
public static void main(String[] args) {
System.out.println(
"Entering first try block");
try {
System.out.println(
"Entering second try block");
try {
throw new Ex();
} finally {
System.out.println(
"finally in 2nd try block");
}
} catch(Ex e) {
System.out.println(
"Caught Ex in first try block");
} finally {
System.out.println(
"finally in 1st try block");
}
}
} ///:~
该措施的输出展示了详细产生的工作:
Entering first try block Entering second try block finally in 2nd try block Caught Ex in first try block finally in 1st try block
若挪用了break和continue语句,finally语句也会得以执行。请留意,与作上标签的break和continue一道,finally解除了Java对goto跳转语句的需求。