垃圾回收与对象的引用

垃圾回收

当程序创建对象、数组等引用类型实体时,系统就会在对内存中为之分配一块内存区,对象就保存在这块内存区中,当这块内存不再被任何引用变量引用时,这块内存就变成垃圾,等待垃圾回收机制进行回收。

垃圾回收机制具有如下特点:

  • 垃圾回收机制只负责回收堆内存中的对象,不会回收任何物理资源。
  • 程序无法精确控制垃圾回收的运行,垃圾回收会在合适的时候进行。当对象永久性地失去引用后,系统就会在合适的时候回收它所占的内存。
  • 在垃圾回收机制回收任何对象之前,总会调用它的finalize()方法,可以重写该方法,让一个引用变量重新引用该对象,从而导致垃圾回收机制取消回收。

对象在内存中的状态

  • 可达状态:当一个对象创建后,若有变量引用它,则这个对象在程序中处于可达状态。
  • 可恢复状态:如果程序中某个对象不再有任何引用变量应用它,且该对象还没有调用finalize()方法。在这种状态下,系统的垃圾回收机制准备回收该对象所引用的内存,在回收该对象之前,系统会调用所有可恢复状态对象的finalize()方法进行资源清理。
  • 不可达状态:当对象与所有引用变量的关联都被切断,且系统已经调用所有对象的finalize()方法后依然没有使该对象变为可达状态,那么该对象将永久地失去应用,变成不可达状态,系统会回收处于改状态的资源。

强制垃圾回收

当一个对象失去引用后,系统何时调用它的finalize()方法对它进行资源清理,对于程序是不可控的。
程序无法精确控制Java垃圾回收的时机,但依然可以强制系统进行垃圾回收-这种强制只是通知系统进行垃圾回收,但系统是否进行垃圾回收依然不确定。大部分时候,程序强制系统垃圾回收后总会有一些效果。强制系统垃圾回收有如下两种方式:

  • System.gc()
  • Runtime.getRuntime().gc()

finalize方法

Object类的finalize()方法没有做任何事情,子类可以选择重写,如下:

protected void finalize() throws Throwable { }

下面我重写finalize()方法,使一个没被应用的对象重新被引用,变成可达状态。

/**
 * Created by SqMax on 2018/5/12.
 */
public class FinalizeTest {
    private static FinalizeTest ft;
    public void info(){
        System.out.println("测试方法");
    }

    @Override
    protected void finalize() throws Throwable {
        System.out.println("让变量引用该对象,使其变为可达状态");
        ft=this;
    }

    public static void main(String[] args) {
//        该对象没有任何引用变量引用,处于可恢复状态
        new FinalizeTest();
        System.gc();
//        Runtime.getRuntime().gc();
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        ft.info();
    }
}

上面我们调用System.gc()方法后,还让当前线程休眠了1秒钟,因为垃圾回收机制也是一个线程,我们通知垃圾回收机制回收垃圾后,垃圾回收线程还是和main线程争夺CPU资源,所以让main线程休眠,这样垃圾回收线程就会通知没被引用的对象调用finalize()方法,是我们的ft变量引用这个对象。
执行结果如下:

让变量引用该对象,使其变为可达状态
测试方法

finalize方法有如下4个特点:

  • 永远不要调用某个对象的finalize()方法,该方法应交给垃圾回收机制调用。
  • finalize()方法何时调用具有不确定性,不要把finalize()方法当成一定被执行的方法。
  • 当JVM执行finalize()方法出现异常时,垃圾回收机制不会报告异常,程序继续执行。
  • 垃圾回收机制是一个后台线程,finalize()方法也是在这个后台线程里调用的,主线程执行完后,这个后台线程也结束,如下:
public class FinalizeTest {
    private static FinalizeTest ft;
    public void info(){
        System.out.println("测试方法");
    }
    @Override
    protected void finalize() throws Throwable {
        System.out.println("正在进行垃圾回收....");
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("让变量引用该对象,使其变为可达状态");
        ft=this;
    }

    public static void main(String[] args) {
//        该对象没有任何引用变量引用,处于可恢复状态
        new FinalizeTest();
        System.gc();
    }
}

执行结果如下:

正在进行垃圾回收....

可以看到finalize()方法还没运行完,程序就结束了。

由于finalize()方法并不一定会被执行,而且不一定会执行完,因此如果清理某个类里打开的资源,则不要放在finalize()方法里进行清理,而应该在finally异常处理的finally块里清理。

对象的软、弱、和虚引用

目前还没有在项目中遇到该知识点的应用,以后遇到在补充.........

原文地址:https://www.cnblogs.com/sqmax/p/9045394.html

时间: 2024-10-17 18:18:44

垃圾回收与对象的引用的相关文章

java中垃圾回收机制中的引用计数法和可达性分析法(最详细)

首先,我这是抄写过来的,写得真的很好很好,是我看过关于GC方面讲解最清楚明白的一篇.原文地址是:https://www.zhihu.com/question/21539353 原文地址:https://www.cnblogs.com/igoodful/p/8727241.html

Java 垃圾回收机制(早期版本)

Java 垃圾回收机制在我们普通理解来看,应该视为一种低优先级的后台进程来实现的,其实早期版本的Java虚拟机并非以这种方式实现的. 先从一种很简单的垃圾回收方式开始. 引用计数 引用计数是一种简单但是速度很慢的垃圾回收技术. 每个对象都含有要给引用计数器,当有引用连接至对象时,引用计数+1. 当引用离开作用域或者被置为null时,引用计数-1. 当发现某个对象的引用计数为0时,就释放其占用的空间.   这种方法开销在整个程序生命周期中持续发生,并且该方法有个缺陷,如果对象之间存在循环引用,可能

Java内存管理及垃圾回收总结

概述 Java和C++的一个很重要的差别在于对内存的管理.Java的自己主动内存管理及垃圾回收技术使得Java程序猿不须要释放废弃对象的内存.从而简化了编程的过程.同一时候也避免了因程序猿的疏漏而导致的内存泄露问题. 内存管理和垃圾回收是JVM很重要的一个部分.深入理解Java的内存管理和垃圾回收机制是避免及修复Java相关异常(OutOfMemoryError, StackOverflowError),理解Java对象创建过程,有效利用内存.构建高性能Java应用的前提.本文将先后介绍Java

【JVM】JVM系列之垃圾回收(二)

一.为什么需要垃圾回收 如果不进行垃圾回收,内存迟早都会被消耗空,因为我们在不断的分配内存空间而不进行回收.除非内存无限大,我们可以任性的分配而不回收,但是事实并非如此.所以,垃圾回收是必须的. 二.哪些内存需要进行垃圾回收 对于虚拟机中线程私有的区域,如程序计数器.虚拟机栈.本地方法栈都不需要进行垃圾回收,因为它们是自动进行的,随着线程的消亡而消亡,不需要我们去回收,比如栈的栈帧结构,当进入一个方法时,就会产生一个栈帧,栈帧大小也可以借助类信息确定,然后栈帧入栈,执行方法体,退出方法时,栈帧出

垃圾回收算法(5)分代回收

分代垃圾回收,基于的是“大部分的对象,在生成后马上就会变成垃圾”这一经验上的事实为设计出发点.此前讨论过基于引事实的另一个垃圾回收算法,引用计数出的一些优化思路. 分代的关键是: 给对象记录下一个age,随着每一次垃圾回收,这个age会增加: 给不同age的对象分配不同的堆内内存空间,称为某一代: 对某一代的空间,有适合其的垃圾回收算法: 对每代进行不同垃圾回收,一般会需要一个额外的信息:即每代中对象被其他代中对象引用的信息.这个引用信息对于当前代来说,扮演与"root"一样的角色,也

Java深度历险(四)——Java垃圾回收机制与引用类型

Java语言的一个重要特性是引入了自动的内存管理机制,使得开发人员不用自己来管理应用中的内存.C/C++开发人员需要通过malloc/free 和new/delete等函数来显式的分配和释放内存.这对开发人员提出了比较高的要求,容易造成内存访问错误和内存泄露等问题.一个常见的问题是会产生“悬挂引用(dangling references)”,即一个对象引用所指向的内存区块已经被错误的回收并重新分配给新的对象了,程序如果继续使用这个引用的话会造成不可预期的结果.开发人员有可能忘记显式的调用释放内存

垃圾回收GC:.Net自动内存管理 上(三)终结器

垃圾回收GC:.Net自动内存管理 上(三)终结器 垃圾回收GC:.Net自动内存管理 上(一)内存分配 垃圾回收GC:.Net自动内存管理 上(二)内存算法 垃圾回收GC:.Net自动内存管理 上(三)终结器 前言 .Net下的GC完全解决了开发者跟踪内存使用以及控制释放内存的窘态.然而,你或午想要理解GC是怎么工作的.此系列文章中将会解释内存资源是怎么被合理分配及管理的,并包含非常详细的内在算法描述.同时,还将讨论GC的内存清理流程及什么时清理,怎么样强制清理. 终结器 GC提供了另外一个能

4.5-全栈Java笔记:垃圾回收机制

垃圾回收机制(Garbage  Collection) Java引入了垃圾回收机制,令C++程序员最头疼的内存管理问题迎刃而解.JAVA程序员可以将更多的精力放到业务逻辑上而不是内存管理工作上,大大的提高了开发效率. 垃圾回收原理和算法 1)内存管理 Java的内存管理很大程度指的就是对象的管理,其中包括对象空间的分配和释放. 对象空间的分配:使用new关键字创建对象即可 对象空间的释放:将对象赋值null即可.垃圾回收器将负责回收所有"不可达"对象的内存空间. 2)垃圾回收过程 任何

<转> python的垃圾回收机制

Python的GC模块主要运用了"引用计数"(reference counting)来跟踪和回收垃圾.在引用计数的基础上,还可以通过"标记-清除"(mark and sweep)解决容器对象可能产生的循环引用的问题.通过"分代回收"(generation collection)以空间换取时间来进一步提高垃圾回收的效率. 引用计数机制: python里每一个东西都是对象,它们的核心就是一个结构体:PyObject 1 typedef struct_