Finally子句的用法
2004-07-21 10:33
495 查看
Finally子句的用法<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />
与其它模型语言相比,finally关键字是对Java异常处理模型的最佳补充。Finally结构使代码总会执行,而不管有无异常发生。
Java虚拟机在每个try语句块和与其相关的catch子句的结尾处都会调用finally子句的子例程。如果没有抛出异常,或执行return、continue、break等情况,隶属于这个finally子句的子例程执行“返回”操作。程序在第一次调用微型子例程的地方继续执行后面的语句。
Jsr和jsr_w指令是使Java虚拟机跳转到微型子例程的操作码。区别是前一个支持双字节的操作数,后一个支持4个字节的操作数。当Java虚拟机遇到jsr或者jsr_w指令,它会把返回地址压入栈,然后从微型子例程的开始处继续执行。
微型子例程执行完毕后,将调用ret指令。Ret指令的功能是执行从子例程中返回的操作。但是如果在子例程中使用了break、return、continue,或者抛出异常退出finally子句,这个ret指令将不会被执行到。正因为如此,Java虚拟机从finally子句开始的地方就将返回地址从栈顶弹出,并且存储在局部变量中。
import java.net.*;
import java.io.*;
class WithoutFinally
{
public void foo() throws IOException
{
//在任一个空闲的端口上创建一个套接字
ServerSocket ss = new ServerSocket(0);
try {
Socket socket = ss.accept();
//此处的其他代码...
}
catch (IOException e) {
ss.close(); //1
throw e;
}
//...
ss.close(); //2
}
}
这段代码创建了一个套接字,并调用 accept 方法。在退出该方法之前,您必须关闭此套接字,以避免资源漏洞。为了完成这一任务,我们在 //2 处调用 close,它是该方法的最后一条语句。但是,如果 try 块中发生一个异常会怎么样呢?在这种情况下,//2 处的 close 调用永远不会发生。因此,您必须捕获这个异常,并在重新发出这个异常之前在 //1 处插入对 close 的另一个调用。这样就可以确保在退出该方法之前关闭套接字。
这样编写代码既麻烦又易于出错,但在没有 finally 的情况下这是必不可少的。不幸的是,在没有 finally 机制的语言中,程序员就可能忘记以这种方式组织他们的代码,从而导致资源漏洞。Java 中的 finally 子句解决了这个问题。有了 finally,前面的代码就可以重写为以下的形式:
import java.net.*;
import java.io.*;
class WithFinally
{
public void foo2() throws IOException
{
//在任一个空闲的端口上创建一个套接字
ServerSocket ss = new ServerSocket(0);
try {
Socket socket = 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 也不会执行。)
与其它模型语言相比,finally关键字是对Java异常处理模型的最佳补充。Finally结构使代码总会执行,而不管有无异常发生。
Java虚拟机在每个try语句块和与其相关的catch子句的结尾处都会调用finally子句的子例程。如果没有抛出异常,或执行return、continue、break等情况,隶属于这个finally子句的子例程执行“返回”操作。程序在第一次调用微型子例程的地方继续执行后面的语句。
Jsr和jsr_w指令是使Java虚拟机跳转到微型子例程的操作码。区别是前一个支持双字节的操作数,后一个支持4个字节的操作数。当Java虚拟机遇到jsr或者jsr_w指令,它会把返回地址压入栈,然后从微型子例程的开始处继续执行。
微型子例程执行完毕后,将调用ret指令。Ret指令的功能是执行从子例程中返回的操作。但是如果在子例程中使用了break、return、continue,或者抛出异常退出finally子句,这个ret指令将不会被执行到。正因为如此,Java虚拟机从finally子句开始的地方就将返回地址从栈顶弹出,并且存储在局部变量中。
import java.net.*;
import java.io.*;
class WithoutFinally
{
public void foo() throws IOException
{
//在任一个空闲的端口上创建一个套接字
ServerSocket ss = new ServerSocket(0);
try {
Socket socket = ss.accept();
//此处的其他代码...
}
catch (IOException e) {
ss.close(); //1
throw e;
}
//...
ss.close(); //2
}
}
这段代码创建了一个套接字,并调用 accept 方法。在退出该方法之前,您必须关闭此套接字,以避免资源漏洞。为了完成这一任务,我们在 //2 处调用 close,它是该方法的最后一条语句。但是,如果 try 块中发生一个异常会怎么样呢?在这种情况下,//2 处的 close 调用永远不会发生。因此,您必须捕获这个异常,并在重新发出这个异常之前在 //1 处插入对 close 的另一个调用。这样就可以确保在退出该方法之前关闭套接字。
这样编写代码既麻烦又易于出错,但在没有 finally 的情况下这是必不可少的。不幸的是,在没有 finally 机制的语言中,程序员就可能忘记以这种方式组织他们的代码,从而导致资源漏洞。Java 中的 finally 子句解决了这个问题。有了 finally,前面的代码就可以重写为以下的形式:
import java.net.*;
import java.io.*;
class WithFinally
{
public void foo2() throws IOException
{
//在任一个空闲的端口上创建一个套接字
ServerSocket ss = new ServerSocket(0);
try {
Socket socket = 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 也不会执行。)
相关文章推荐
- Java实现时间日期格式转换示例
- Java下利用Jackson进行JSON解析和序列化示例
- Java的几个重要版本_动力节点Java学院整理
- Java Web开发之信息查询方式总结
- 基于Java字符编码的使用详解
- java获取Date时间的各种方式汇总
- Java使用AES加密和解密的实例详解
- 基于java时区转换夏令时的问题及解决方法
- 转 -- Java 学习方法浅谈 --来自java视线 Robbin
- ATL实现Connection Point的一种简单的方法
- 我的应聘总结!
- Java测试规范(引用)
- IP地址的隐藏(delphi实现 )
- DirectX.Capture
- 开发工具浅谈
- Java日记(1) 一. 认识Java
- 使用C#开发COM+组件
- Java的中文编程与配置心得
- 纯编码实现数据库的建立或压缩