java中的finalize()

Java的垃圾回收器在回收某个对象的内存之前,会调用finalize()方法进行资源清理。
如果程序在终止之前始终没有进行垃圾回收,那么对象的finalize()是始终不会调用的。
    关于finalize()我们有以下几点说明:

  • 1.永远不要主动的去掉用对象的finalize(),交给垃圾回收机制去调用
  • 2.finalize()是否被调用和何时被调用都有不确定性
  • 3.如果JVM执行可恢复对象的finalize(),可能会使得该对象重新变成可达状态
  • 4.如果JVM执行finalize方法时出现异常,垃圾回收机制不会报告异常,程序继续执行

下面给出一段测试代码,来演示对象从可恢复状态转换成可达状态的情况。

public class StringDemo {
	private static StringDemo demo = null;
	public static void main(String[] args) {
		System.out.println("new=" + new StringDemo());
		System.out.println("new=" + new StringDemo());
		System.out.println("new=" + new StringDemo());
		System.gc();
		System.runFinalization();
		demo.info();
	}
	public void info() {
		System.out.println("-----info-----");
	}
	@Override
	protected void finalize() throws Throwable {
		demo = this;
		System.out.println("this=" + this);
		System.out.println("demo=" + demo);
	}
}  

在主函数中,我定义了三个StringDemo匿名对象,这三个对象都是在堆内存里面,没有任何引用变量指向它们,也就是一初始化之后就处于可恢复状态。在这之后,调用System.gc(),通知JVM该收垃圾了,但是在下面紧接着调用了System.runFinalization(),这个方法会强制垃圾回收期调用系统中可恢复对象的finalize(),因此,在被回收之前,调用了每个匿名对象的finalize()方法。在finalize()里面我做了什么呢?我把一个静态的StringDemo变量指向了自身,这样demo就不是null了,所以在demo.info();方法的时候,可以正常输出,也就是转换成了可达状态,然后程序结束了。

如果不添加System.runFinalization();那么demo就是null,程序就会空指针异常。

下面是程序运行的结果

时间: 2024-08-30 07:08:42

java中的finalize()的相关文章

JAVA中的finalize()方法

[转]JAVA中的finalize()方法 今天早上看Thinking in java的[第四章 初始化和清除].[  清除:终结和垃圾回收]的时候, 看到了这个东西. 用于清理滴... 当然,这个方法来自java.lang.Object finalize()方法的重写 权限(Access)需要是protected或者是public ,不能是private finalize()方法不需要显示地调用, 在垃圾回收(GC)时会被自动先行调用的. 据我测试,需要显示地调用垃圾回收方法(System.g

Java中的finalize方法理解

首先:system.gc()并不是你调用就马上执行的, 而是根据虚拟机的各种算法来来计算出执行垃圾回收的时间,另外,程序自动结束时不会执行垃圾回收的. 其次:对象被回收时,要经过两次标记,第一次标记,如果finalize被重写,或者finalize被调用过,那么垃圾回收并不会去执行finalize,第二次标记,如果对象不能在finalize中成功拯救自己,那真的就要被回收了. 咱们用的虚拟机一般都是sun的hotspot,以下描述都是针对hotspot虚拟机,当然其他虚拟机也差不多, ,,他是根

夯实Java基础系列10:深入理解Java中的异常体系

目录 为什么要使用异常 异常基本定义 异常体系 初识异常 异常和错误 异常的处理方式 "不负责任"的throws 纠结的finally throw : JRE也使用的关键字 异常调用链 自定义异常 异常的注意事项 当finally遇上return JAVA异常常见面试题 参考文章 微信公众号 Java技术江湖 个人公众号:黄小斜 - Java异常 本系列文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查看 https://github.c

Java中final、finally、finalize的区别(转)

Java中final.finally.finalize的区别与用法,困扰了不少学习者,下面我们就这个问题进行一些探讨,希望对大家的学习有所帮助. 方法/步骤 1 简单区别: final用于声明属性,方法和类,分别表示属性不可交变,方法不可覆盖,类不可继承. finally是异常处理语句结构的一部分,表示总是执行. finalize是Object类的一个方法,在垃圾收集器执行的时候会调用被回收对象的此方法,供垃圾收集时的其他资源回收,例如关闭文件等. END 方法/步骤2 1 中等区别: 虽然这三

java中final 、finally、finalize的区别

比较java中常用关键字以免混淆 final :用来修饰变量,表示变量为最终变量,不能被改变 finally:在处理异常的时候使用,表示最终要执行的代码块 finalize:java Object类中的一个方法,用于回收内存和回收前相应的处理,这个方法不鼓励使用

java中的final、finally和finalize

最近在读Thinking In Java,秉着有些地方还能知道自己不会的精神,都去好好查阅了一些资料,在内存分配这一章,看到finalize()这个方法,刚开始很不理解,查阅了一些资料,顺带看了一下final.finally,现在分享一下. 一.final的介绍 final可用在4个地方,分别是变量(static 或者 !static),形式参数,方法和类,每种情况都有不同的含义,下面分别介绍之: final修饰变量: 对基本类型表示变量被赋值后是一个常量,即不可改变的:对引用类型,表示引用指向

JAVA中GC时finalize()方法是不是一定会被执行?

在回答上面问题之前,我们一定要了解JVM在进行垃圾回收时的机制,首先: 一.可达性算法  要知道对象什么时候死亡,我们需要先知道JVM的GC是如何判断对象是可以回收的.JAVA是通过可达性算法来来判断对象是否存活的.这个算法的基本思路就是通过一系列的称为“GC Roots”的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到GC Roots 没有任何引用链相连时,则证明此对象是不可用的. 在JAVA语言中,可以作为GC Roots的对象包括下面几种: * 虚拟机栈(

Java 中的 final、finally、finalize 有什么不同?

Java 中 final.finally.finalize 有什么不同?这是在 Java 面试中经常问到的问题,他们究竟有什么不同呢? 这三个看起来很相似,其实他们的关系就像卡巴斯基和巴基斯坦一样有基巴关系. 那么如果被问到这个问题该怎么回答呢?首先可以从语法和使用角度出发简单介绍三者的不同: final 可以用来修饰类.方法.变量,分别有不同的意义,final 修饰的 class 代表不可以继承扩展,final 的变量是不可以修改的,而 final 的方法也是不可以重写的(override).

java中JVM虚拟机内存模型详细说明

java中JVM虚拟机内存模型详细说明 2012-12-12 18:36:03|  分类: JAVA |  标签:java  jvm  堆内存  虚拟机  |举报|字号 订阅 JVM的内部结构如下图: 一个优秀Java程序员,必须了解Java内存模型.GC工作原理,以及如何优化GC的性能.与GC进行有限的交互,有一些应用程序对性能要求较高,例如嵌入式系统.实时系统等,只有全面提升内存的管理效率,才能提高整个应用程序的性能. 本文将从JVM内存模型.GC工作原理,以及GC的几个关键问题进行探讨,从