讲解Java编程中finally语句的基本使用方法
在java中的finally关键一般与try一起使用,在程序进入try块之后,无论程序是因为异常而中止或其它方式返回终止的,finally块的内容一定会被执行,写个例子来说明下:
publicclassTryAndFinallyTest{
publicstaticvoidmain(String[]args)throwsException{
try{
inta=testFinally(2);
System.out.println("异常返回的结果a:"+a);
}catch(Exceptione){
intb=testFinally(1);
System.out.println("正常返回的结果b:"+b);
}
intb=testFinally(3);
System.out.println("break返回的结果:"+b);
b=testFinally(4);
System.out.println("return返回的结果:"+b);
}
staticinttestFinally(inti)throwsException{
intflag=i;
try{//一旦进去try范围无论程序是抛出异常或其它中断情况,finally的内容都会被执行
switch(i){
case1:++i;break;//程序正常结束
case2:thrownewException("测试下异常情况");
case3:break;
default:return-1;
}
}finally{
System.out.println("finallycomingwheni="+flag);
}
returni;
}
}
执行结果如下
finallycomingwheni=2 finallycomingwheni=1 正常返回的结果b:2 finallycomingwheni=3 break返回的结果:3 finallycomingwheni=4 return返回的结果:-1
结果说明无论上述什么情况,finally块总会被执行。
与其他语言的模型相比,finally关键字是对Java异常处理模型的最佳补充。finally结构使代码总会执行,而不管有无异常发生。使用finally可以维护对象的内部状态,并可以清理非内存资源。如果没有finally,您的代码就会很费解。例如,下面的代码说明,在不使用finally的情况下您必须如何编写代码来释放非内存资源:
importjava.net.*;
importjava.io.*;
classWithoutFinally
{
publicvoidfoo()throwsIOException
{
//在任一个空闲的端口上创建一个套接字
ServerSocketss=newServerSocket(0);
try
{
Socketsocket=ss.accept();
//此处的其他代码...
}
catch(IOExceptione)
{
ss.close();//1
throwe;
}
//...
ss.close();//2
}
}
这段代码创建了一个套接字,并调用accept方法。在退出该方法之前,您必须关闭此套接字,以避免资源漏洞。为了完成这一任务,我们在//2处调用close,它是该方法的最后一条语句。但是,如果try块中发生一个异常会怎么样呢?在这种情况下,//2处的close调用永远不会发生。因此,您必须捕获这个异常,并在重新发出这个异常之前在//1处插入对close的另一个调用。这样就可以确保在退出该方法之前关闭套接字。
这样编写代码既麻烦又易于出错,但在没有finally的情况下这是必不可少的。不幸的是,在没有finally机制的语言中,程序员就可能忘记以这种方式组织他们的代码,从而导致资源漏洞。Java中的finally子句解决了这个问题。有了finally,前面的代码就可以重写为以下的形式:
importjava.net.*;
importjava.io.*;
classWithFinally
{
publicvoidfoo2()throwsIOException
{
//在任一个空闲的端口上创建一个套接字
ServerSocketss=newServerSocket(0);
try
{
Socketsocket=ss.accept();
//此处的其他代码...
}
finally
{
ss.close();
}
}
}
finally块确保close方法总被执行,而不管try块内是否发出异常。因此,可以确保在退出该方法之前总会调用close方法。这样您就可以确信套接字被关闭并且您没有泄漏资源。在此方法中不需要再有一个catch块。在第一个示例中提供catch块只是为了关闭套接字,现在这是通过finally关闭的。如果您确实提供了一个catch块,则finally块中的代码在catch块完成以后执行。
finally块必须与try或try/catch块配合使用。此外,不可能退出try块而不执行其finally块。如果finally块存在,则它总会执行。(无论从那点看,这个陈述都是正确的。有一种方法可以退出try块而不执行finally块。如果代码在try内部执行一条System.exit(0);语句,则应用程序终止而不会执行finally执行。另一方面,如果您在try块执行期间拨掉电源,finally也不会执行。)