- 浏览: 976959 次
- 性别:
- 来自: 上海
文章分类
最新评论
-
Mr.Cheney:
去掉 UUID字符串中的“-” 直接replaceAll(&q ...
JAVA生成全局唯一ID 使用 java.util.UUID -
呜哩喵:
楼主nice
java中的时间操作 -
zxs6587:
Thinking inJava我读着好像说要建立基类对象啊!请 ...
创建子类的对象时也要创建其所有父类的对象? -
just_Word:
getFullYear
date.getyear -
JamesQian:
我觉得楼上的synchronized(this),notify ...
notify() wait()
当垃圾回收器将要释放无用对象的内存时,先调用该对象的finalize()方法。如果在程序终止之前垃圾回收器始终没有执行垃圾回收操作,那么垃圾回收器将始终不会调用无用对象的finalize()方法。在Java的Object祖先类中提供了protected 类型的finalize()方法,因此任何Java类都可以覆盖finalize()方法,在这个方法中进行释放对象所占的相关资源的操作。
Java虚拟机的垃圾回收操作对程序完全是透明的,因此程序无法预料某个无用对象的finalize()方法何时被调用。另外,除非垃圾回收器认为程序需要额外的内存,否则它不会试图释放无用对象占用的内存。换句话说,以下情况是完全可能的:一个程序只占用了少量内存,没有造成严重的内存需求,于是垃圾回收器没有释放那些无用对象占用的内存,因此这些对象的finalize()方法还没有被调用,程序就终止了。
程序即使显式调用System.gc()或Runtime.gc()方法,也不能保证垃圾回收操作一定执行,因此不能保证无用对象的finalize()方法一定被调用。
11.4.4 对象的finalize()方法的特点
对象的finalize()方法具有以下特点:
垃圾回收器是否会执行该方法及何时执行该方法,都是不确定的。
finalize()方法有可能使对象复活,使它恢复到可触及状态。
垃圾回收器在执行finalize()方法时,如果出现异常,垃圾回收器不会报告异常,程序继续正常运行。
下面结合一个具体的例子来解释finalize()方法的特点。例程11-13的Ghost类是一个带实例缓存的不可变类,它的finalize()方法能够把当前实例重新加入到实例缓存ghosts中。
例程11-13 Ghost.java
import java.util.Map;
import java.util.HashMap;
public class Ghost {
private static final Map<String,Ghost> ghosts=new HashMap<String,Ghost>();
private final String name;
public Ghost(String name) {
this.name=name;
}
public String getName(){return name;}
public static Ghost getInstance(String name){
Ghost ghost =ghosts.get(name);
if (ghost == null) {
ghost=new Ghost(name);
ghosts.put(name,ghost);
}
return ghost;
}
public static void removeInstance(String name){
ghosts.remove(name);
}
protected void finalize()throws Throwable{
ghosts.put(name,this);
System.out.println("execute finalize");
//throw new Exception("Just Test");
}
public static void main(String args[])throws Exception{
Ghost ghost=Ghost.getInstance("IAmBack"); //①
System.out.println(ghost); //②
String name=ghost.getName(); //③
ghost=null; //④
Ghost.removeInstance(name); //⑤
System.gc(); //⑥
//把CPU让给垃圾回收线程
Thread.sleep(3000); //⑦
ghost=Ghost.getInstance("IAmBack"); //⑧
System.out.println(ghost); //⑨
}
}
运行以上Ghost类的main()方法,一种可能的打印结果为:
Ghost@3179c3
execute finalize
Ghost@3179c3
以上程序创建了3个对象:1个Ghost对象、1个常量字符串“IAmBack”及1个HashMap对象。当程序执行完main()方法的第③行时,内存中引用变量与对象之间的关系如图11-9所示。
图11-9 Ghost对象与其他对象及引用变量的关系
当执行完第④行时,ghost变量被置为null,此时Ghost对象依然被ghosts属性间接引用,因此仍然处于可触及状态。当执行完第⑤行时,Ghost对象的引用从HashMap对象中删除,Ghost对象不再被程序引用,此时进入可复活状态,即变为无用对象。
第⑥行调用System.gc()方法,它能提高垃圾回收器尽快执行垃圾回收操作的可能性。假如垃圾回收器线程此刻获得了对CPU的使用权,它将调用Ghost对象的finalize()方法。该方法把Ghost对象的引用又加入到HashMap对象中,Ghost对象又回到可触及状态,垃圾回收器放弃回收它的内存。执行完第⑧行,ghost变量又引用这个Ghost对象。
假如对finalize()做一些修改,使它抛出一个异常:
protected void finalize()throws Throwable{
ghosts.put(name,this);
System.out.println("execute finalize");
throw new Exception("Just Test");
}
程序的打印结果不变。由此可见,当垃圾回收器执行finalize()方法时,如果出现异常,垃圾回收器不会报告异常,也不会导致程序异常中断。
假如在程序运行中,垃圾回收器始终没有执行垃圾回收操作,那么Ghost对象的finalize()方法就不会被调用。读者不妨把第⑥行的System.gc()和第⑦行的Thread.sleep(3000)方法注释掉,这样更加可能导致finalize()方法不会被调用,此时程序的一种可能的打印结果为:
Ghost@3179c3
Ghost@310d42
从以上打印结果可以看出,由于Ghost对象的finalize()方法没有被执行,因此这个Ghost对象在程序运行期间始终没有复活。当程序第二次调用Ghost.getInstance("IAmBack")方法时,该方法创建了一个新的Ghost对象。
值得注意的是,以上例子仅仅用于演示finalize()方法的特性,在实际应用中,不提倡用finalize() 方法来复活对象。可以把处于可触及状态的对象比做活在阳间的人,把不处于这个状态的对象(无用对象)比做到了阴间的人。程序所能看见和使用的是阳间的人,假如阎王经常悄悄地让几个阴间的人复活,使他们在程序毫不知情的情况下溜回阳间,这只会扰乱程序的正常执行流程。
11.4.5 比较finalize()方法和finally代码块
在Object类中提供了finalize()方法,它的初衷是用于在对象被垃圾回收器回收之前,释放所占用的相关资源,这和try…catch…finally语句的finally代码块的用途比较相似。但由于垃圾回收器是否会执行finalize()方法及何时执行该方法,都是不确定的,因此在程序中不能用finalize()方法来完成同时具有以下两个特点的释放资源的操作。
必须执行。
必须在某个确定的时刻执行。
具有以上特点的操作更适合于放在finally代码块中。此外,可以在类中专门提供一个用于释放资源的公共方法,最典型的就是java.io.InputStream和java.io.OutputStream类的close()方法,它们用于关闭输入流或输出流。当程序中使用了一个输入流时,在结束使用前应该确保关闭输入流。
InputStream in;
try{
InputStream in=new FileInputStream("a.txt");
…
}catch(IOException e){
…
}finally{
try{in.close();}catch(IOException e){…}
}
在多数情况下,应该避免使用finalize()方法,因为它会导致程序运行结果的不确定性。在某些情况下,finalize()方法可用来充当第二层安全保护网,当用户忘记显式释放相关资源时,finalize()方法可以完成这一收尾工作。尽管 finalize()方法不一定会被执行,但是有可能会释放资源,这总比永远不会释放资源更安全。
可以用自动洗衣机的关机功能来解释finalize()方法的用途。自动洗衣机向用户提供了专门的关机按钮,这相当于AutoWasher类的close()方法,假如用户忘记关机,相当于忘记调用AutoWasher对象的close()方法,那么自动洗衣机会在洗衣机停止工作后的1个小时内自动关机,这相当于调用finalize()方法。当然,这个例子不是太贴切,因为如果用户忘记关机,洗衣机的自动关机操作总会被执行。
Java虚拟机的垃圾回收操作对程序完全是透明的,因此程序无法预料某个无用对象的finalize()方法何时被调用。另外,除非垃圾回收器认为程序需要额外的内存,否则它不会试图释放无用对象占用的内存。换句话说,以下情况是完全可能的:一个程序只占用了少量内存,没有造成严重的内存需求,于是垃圾回收器没有释放那些无用对象占用的内存,因此这些对象的finalize()方法还没有被调用,程序就终止了。
程序即使显式调用System.gc()或Runtime.gc()方法,也不能保证垃圾回收操作一定执行,因此不能保证无用对象的finalize()方法一定被调用。
11.4.4 对象的finalize()方法的特点
对象的finalize()方法具有以下特点:
垃圾回收器是否会执行该方法及何时执行该方法,都是不确定的。
finalize()方法有可能使对象复活,使它恢复到可触及状态。
垃圾回收器在执行finalize()方法时,如果出现异常,垃圾回收器不会报告异常,程序继续正常运行。
下面结合一个具体的例子来解释finalize()方法的特点。例程11-13的Ghost类是一个带实例缓存的不可变类,它的finalize()方法能够把当前实例重新加入到实例缓存ghosts中。
例程11-13 Ghost.java
import java.util.Map;
import java.util.HashMap;
public class Ghost {
private static final Map<String,Ghost> ghosts=new HashMap<String,Ghost>();
private final String name;
public Ghost(String name) {
this.name=name;
}
public String getName(){return name;}
public static Ghost getInstance(String name){
Ghost ghost =ghosts.get(name);
if (ghost == null) {
ghost=new Ghost(name);
ghosts.put(name,ghost);
}
return ghost;
}
public static void removeInstance(String name){
ghosts.remove(name);
}
protected void finalize()throws Throwable{
ghosts.put(name,this);
System.out.println("execute finalize");
//throw new Exception("Just Test");
}
public static void main(String args[])throws Exception{
Ghost ghost=Ghost.getInstance("IAmBack"); //①
System.out.println(ghost); //②
String name=ghost.getName(); //③
ghost=null; //④
Ghost.removeInstance(name); //⑤
System.gc(); //⑥
//把CPU让给垃圾回收线程
Thread.sleep(3000); //⑦
ghost=Ghost.getInstance("IAmBack"); //⑧
System.out.println(ghost); //⑨
}
}
运行以上Ghost类的main()方法,一种可能的打印结果为:
Ghost@3179c3
execute finalize
Ghost@3179c3
以上程序创建了3个对象:1个Ghost对象、1个常量字符串“IAmBack”及1个HashMap对象。当程序执行完main()方法的第③行时,内存中引用变量与对象之间的关系如图11-9所示。
图11-9 Ghost对象与其他对象及引用变量的关系
当执行完第④行时,ghost变量被置为null,此时Ghost对象依然被ghosts属性间接引用,因此仍然处于可触及状态。当执行完第⑤行时,Ghost对象的引用从HashMap对象中删除,Ghost对象不再被程序引用,此时进入可复活状态,即变为无用对象。
第⑥行调用System.gc()方法,它能提高垃圾回收器尽快执行垃圾回收操作的可能性。假如垃圾回收器线程此刻获得了对CPU的使用权,它将调用Ghost对象的finalize()方法。该方法把Ghost对象的引用又加入到HashMap对象中,Ghost对象又回到可触及状态,垃圾回收器放弃回收它的内存。执行完第⑧行,ghost变量又引用这个Ghost对象。
假如对finalize()做一些修改,使它抛出一个异常:
protected void finalize()throws Throwable{
ghosts.put(name,this);
System.out.println("execute finalize");
throw new Exception("Just Test");
}
程序的打印结果不变。由此可见,当垃圾回收器执行finalize()方法时,如果出现异常,垃圾回收器不会报告异常,也不会导致程序异常中断。
假如在程序运行中,垃圾回收器始终没有执行垃圾回收操作,那么Ghost对象的finalize()方法就不会被调用。读者不妨把第⑥行的System.gc()和第⑦行的Thread.sleep(3000)方法注释掉,这样更加可能导致finalize()方法不会被调用,此时程序的一种可能的打印结果为:
Ghost@3179c3
Ghost@310d42
从以上打印结果可以看出,由于Ghost对象的finalize()方法没有被执行,因此这个Ghost对象在程序运行期间始终没有复活。当程序第二次调用Ghost.getInstance("IAmBack")方法时,该方法创建了一个新的Ghost对象。
值得注意的是,以上例子仅仅用于演示finalize()方法的特性,在实际应用中,不提倡用finalize() 方法来复活对象。可以把处于可触及状态的对象比做活在阳间的人,把不处于这个状态的对象(无用对象)比做到了阴间的人。程序所能看见和使用的是阳间的人,假如阎王经常悄悄地让几个阴间的人复活,使他们在程序毫不知情的情况下溜回阳间,这只会扰乱程序的正常执行流程。
11.4.5 比较finalize()方法和finally代码块
在Object类中提供了finalize()方法,它的初衷是用于在对象被垃圾回收器回收之前,释放所占用的相关资源,这和try…catch…finally语句的finally代码块的用途比较相似。但由于垃圾回收器是否会执行finalize()方法及何时执行该方法,都是不确定的,因此在程序中不能用finalize()方法来完成同时具有以下两个特点的释放资源的操作。
必须执行。
必须在某个确定的时刻执行。
具有以上特点的操作更适合于放在finally代码块中。此外,可以在类中专门提供一个用于释放资源的公共方法,最典型的就是java.io.InputStream和java.io.OutputStream类的close()方法,它们用于关闭输入流或输出流。当程序中使用了一个输入流时,在结束使用前应该确保关闭输入流。
InputStream in;
try{
InputStream in=new FileInputStream("a.txt");
…
}catch(IOException e){
…
}finally{
try{in.close();}catch(IOException e){…}
}
在多数情况下,应该避免使用finalize()方法,因为它会导致程序运行结果的不确定性。在某些情况下,finalize()方法可用来充当第二层安全保护网,当用户忘记显式释放相关资源时,finalize()方法可以完成这一收尾工作。尽管 finalize()方法不一定会被执行,但是有可能会释放资源,这总比永远不会释放资源更安全。
可以用自动洗衣机的关机功能来解释finalize()方法的用途。自动洗衣机向用户提供了专门的关机按钮,这相当于AutoWasher类的close()方法,假如用户忘记关机,相当于忘记调用AutoWasher对象的close()方法,那么自动洗衣机会在洗衣机停止工作后的1个小时内自动关机,这相当于调用finalize()方法。当然,这个例子不是太贴切,因为如果用户忘记关机,洗衣机的自动关机操作总会被执行。
发表评论
-
关于数组和List之间相互转换的方法
2011-04-14 21:04 13801.List转换成为数组。( ... -
java的几种对象(PO,VO,DAO,BO,POJO)解释
2011-03-24 10:13 1338java的几种对象(PO,VO,DAO,BO,POJO)解释 ... -
switch
2010-12-02 19:02 11431 public class Switch { 2 ... -
优化的冒泡排序
2010-09-25 14:18 1364public static void bubble_Sort( ... -
java变量命名规则
2010-08-13 23:15 23661. 大小写有别,例如 a 和 A是两个变量 2. 长度任意 ... -
String.getBytes()的问题
2010-08-13 22:46 1643转载 http://java.chinaitlab.c ... -
tomcat 修改端口
2010-08-09 22:41 2001Tomcat端口修改: 在Tomcat安装目录下的conf目 ... -
tomcat 中增加用户名和密码
2010-08-09 22:41 1911原来的tomcat-user.xml是 <?xml ... -
Eclipse is running in a JRE, but a JDK is required
2010-07-28 09:30 15211 安装了maven插件,使用的时候老是有这样的提示: 08- ... -
安装Eclipse的maven插件
2010-07-27 11:01 1817Installing m2eclipse Core To i ... -
Attach Library Sources and Javadocs
2010-07-26 13:41 1911Attach Library Sources and Java ... -
maven 安装jaxb插件
2010-07-18 15:10 65591. Put your schemas ( ... -
java接受控制台输入
2010-07-16 13:45 2708import java.io.*; public c ... -
将xsd文件转化为java类
2010-07-10 15:31 2509最近有一个需求是把xsd文件中定义的数据类型转化为java类 ... -
jconsole attache sun glassfish
2010-06-13 17:04 1348To Set Up JConsole Connectivity ... -
suse下lamp的安装
2010-05-31 16:45 1561首先卸载suse缺省安装的apache2 主要是在网上看到人家 ... -
java的property配置文件的用法
2010-05-30 15:04 1141在我们平时写程序的时候,有些参数是经常改变的,而这种改变不是我 ... -
让ubuntu下的eclipse支持GBK编码
2010-05-30 14:38 1511今天,把windows下的工程导入到了Linux下eclips ... -
java路径中/的问题
2010-05-18 17:23 1361windows支持两种文件分隔符“/”和“\” 且使用“/”时 ... -
java中serializable是可以继承的
2010-05-16 21:58 5505import java.io.FileInputStream; ...
相关推荐
在 Java 中,finalize() 方法是 Object 类的一个方法,默认情况下,纯 Java 编写的 Class 不需要重新覆盖这个方法。只有在特殊情况下,例如使用 JNI 或 C++ 编写的 Class 时,需要覆盖默认的 finalize() 方法,以...
Java 中 finalize 方法使用 finalize 方法是 Java 编程语言中一个非常重要的方法,它与 Java 编程中的垃圾回收器有着紧密的关系。当一个对象变成一个垃圾对象的时候,如果此对象的内存被回收,那么就可以调用系统中...
Java中的`finalize()`方法是Java语言提供的一种机制,用于在对象被垃圾回收之前进行资源清理。这个方法在Java的`Object`类中被定义为`protected`类型,因此所有Java类都能够覆盖它来实现特定的清理操作。在垃圾回收...
Java禁止使用finalize方法 Java中的finalize方法是一个特殊的方法,位于Java.lang.Object类中,它的主要作用是允许开发者在对象被垃圾回收前执行一些清理工作。但是,finalize方法存在许多问题,导致它不被推荐使用...
在Java编程语言中,`finalize()`方法是一个特殊的方法,它属于`Object`类,主要用于对象销毁前执行一些必要的清理工作。然而,随着Java版本的更新和垃圾收集机制的改进,`finalize()`方法逐渐失去了其原本的重要性,...
Java中的finalize()方法覆盖实例代码详解 finalize()方法是Java中的一个特殊方法,它可以在对象被垃圾回收器回收之前执行一次。这个方法可以用来释放对象持有的资源,但是它的使用需要非常小心,因为它带有很多不...
Java垃圾回收机制的finalize方法实例分析 Java垃圾回收机制的finalize方法是垃圾回收机制中一个非常重要的概念,它可以帮助开发者更好地理解和掌握垃圾回收机制的工作机理。本文将通过实例形式分析finalize方法的...
这个引用对象专门为带finalize方法的类服务,可以理解为每一个有相应的方法的对象,其都会封装为一种finalRefernece对象. 因为finalize方法是object定义的,其默认实现为空.那么如果重写了此方法,那么方法体...
在Java中,finalize()方法是一个重要的概念,它与final关键字息息相关。下面我们将深入探讨finalize()方法的用法和应用场景。 一、final关键字 final是Java的关键字,它所表示的是“这部分是无法修改的”。在程序...
在Java编程语言中,`final`, `finally`, 和 `finalize` 是三个非常重要的关键字,它们各自扮演着不同的角色,尤其在处理程序的可预测性、数据的不可变性和垃圾回收等方面。下面将详细阐述这三个关键字的区别及其应用...
Java中的`finalize()`方法是对象生命周期的一部分,它与C++中的析构函数类似,但在功能和调用时机上有所不同。`finalize()`方法是在对象被GC标记为可回收并准备释放之前调用的,提供了在对象被彻底删除前执行某些...
Java中的`finalize`方法是Object类的一个特殊方法,它的主要作用是在对象被垃圾收集器回收前执行清理工作。本文将详细解析`finalize`方法的实现和执行过程。 首先,了解`finalize`方法的用途:当一个对象不再被任何...
java.lang.object原始码最终不推荐使用java.lang.Object.finalize() 我的博客文章中的示例的Gradle源代码存储库。 在Linux Mint(SUSE Linux)上构建和测试。 由于gradle当前不支持Java 11,因此可以使用标准Java...
Java9垃圾回收方法finalize() 原理解析 Java9垃圾回收方法中的finalize()方法是一个特殊的方法,它是Object类中的方法,用于在类被GC回收时做一些处理操作。finalize()方法的主要作用是释放对象占用的资源,例如...
finalize() 方法主要用途是回收特殊渠道申请的内存,例如 JNI(Java Native Interface)调用 non-Java 程序(C 或 C++)。 final 关键字用于限制变量、方法和类的修改和继承,finally 块用于异常处理中的清理操作,...
当一个对象被垃圾回收器标记为即将回收时,Java会尝试调用该对象的`finalize`方法,提供一个机会进行必要的清理操作。然而,这个方法并不保证一定会被执行,因此不应该依赖它来清理资源。在实际编程中,`finalize`的...
在Java编程语言中,"final", "finally", 和 "finalize" 这三个关键字有着截然不同的含义和用途,它们各自在程序设计的不同层面扮演着重要角色。深入理解这些概念对于编写高效、健壮的Java代码至关重要。 ### final ...
Java 技术允许使用finalize() 方法在垃圾收集器将对象从内存中清除出去 之前做必要的清理工作。这个方法是由垃圾收集器在确定这个对象没有被引用时对这个对 象调用的。它是在Object 类中定义的,因此所有的类都...