在垃圾回收器中有哪几种判断是否需要被回收的几种方法

垃圾回收器在回收垃圾之前第一件事就是判断哪些是可以被回收的对象,如何判断呢?我们可以根据该对象是否还有引用指向它来进行判断,如果有则不能回收,如果没有则可以回收,具体有以下几种算法:

  1. 引用计数算法:
    给对象添加一个引用计数器,每当它被引用一次计数器就加一,当该引用失效时就减一,如果计数器的值为0就表示它要被垃圾收集器作为垃圾收集了。这种算法存在一个问题,就是如果对象之间互相循环引用,他们就不可能被垃圾回收。
    举个栗子:
    public class RefrenceCounting {
    public Object instance = null;
    }
    public class TestGc {
    public static void main(String[] args) {
    RefrenceCounting rf1 = new RefrenceCounting();
    RefrenceCounting rf2 = new RefrenceCounting();
    rf1.instance = rf2;
    rf2.instance = rf1;
    rf1 = null;
    rf2 = null;
    System.gc();
    }
    }

从图中可以看到,rf1和rf2互相引用,即使rf1和rf2置空,从外界无法访问他们,但是他们的引用计数器不为空,垃圾收集器无法回收他们。

  1. 可达性分析算法:
    通过一系列称为“GC Roots”的对象作为起点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到Gc roots没有任何引用链相连时则此对象是不可用的。

上图中obj1,obj2,obj3都有引用指向Gc Roots,obj4和obj5没有指向Gc Roots的引用,所以他们会作为垃圾回收器的对象。可达性分析算法是java中使用的算法。
可以作为Gc Roots的对象包括以下几种:
1) 虚拟机栈(栈帧中的本地变量表)中引用的对象
2) 方法区中类静态属性引用的对象
3) 方法区中常量引用的对象
4) 本地方法栈中引用的对象

引用计数算法和可达性分析算法都和引用有关,在jdk1.2以后引用被分为四种:

  1. 强引用:
    强引用就是例如 ObjectA a = new ObjectA(),这样的引用,存在这种引用的对象不会被垃圾回收器回收。
    2:软引用:
    软引用是指一些引用还有用但并非必须,被软引用的对象会在内存被占满,即将发生内存溢出异常之前进行回收。
    3:弱引用:被弱引用的对象只能生存到下一次垃圾回收发生之前,当下一次垃圾回收时,无论内存是否被占满都会回收弱引用的对象。
    4:虚引用,被虚引用的对象和没有被引用的对象一样都会被垃圾回收器回收,不同的是,它在被回收时会收到一个系统通知。

原文地址:https://blog.51cto.com/11583017/2468280

时间: 2024-10-13 13:40:48

在垃圾回收器中有哪几种判断是否需要被回收的几种方法的相关文章

JVM:G1垃圾回收器

http://blog.jobbole.com/109170/ https://blog.csdn.net/baiye_xing/article/details/73743395 G1垃圾收集器关键技术. ===G1垃圾回收器是如何划分堆空间的呢? G1垃圾回收器将内存分成一个个小块区域,这些小块区域的大小可以是1M,2M,4M,8M等等,并且这些小块区域可以是eden,survivor,old,以及humongous区域.其中humongous用来存大对象,如果一个对象超过某块空间的1/2的时

总结Java垃圾回收器的方法和原理

1. 垃圾回收只与内存有关 在Java中,我们new完对象之后,垃圾回收器负责回收无用的对象占据的内存资源.这与C++不同,在C++中,准许使用局部对象,回收对象时候,需要用到finalize()析构函数.C++的对象创建在堆栈中,而Java对象创建在堆中,所以我们创建完对象之后,Java的垃圾回收器在堆中,会自动帮我们回收垃圾,至于何时回收垃圾,我们不得而知了. 2.垃圾回收用到的方法 (1)finalize() 该方法是用来回收“特殊”的内存,而这内存不是new出来的,所以垃圾回收器无法回收

第5章(2) Java的垃圾回收器

终结处理 Java有垃圾回收器回收无用对象占据的内存资源.但垃圾回收器只知道释放经由new分配的内存,如果用其他的方法获得了一块"特殊"的内存区域,就需要我们自己完成清理工作.Java提供了finalize()方法来解决这一问题. finalize()方法: 工作原理:一但垃圾回收器准备好释放对象占用的存储空间,将首先调用其finalize()方法,并且在下一次垃圾回收动作发生时,才会真正回收对象占用的内存. 其作用,换句话说就是:在你不再需要某个对象之前,如果必须执行某些动作,那么你

【转帖】CMS垃圾回收器详解

CMS垃圾回收器详解 https://www.jianshu.com/p/08f0b85ad665 垃圾回收器组合 垃圾回收器从线程运行情况分类有三种: 串行回收,Serial回收器,单线程回收,全程stw: 并行回收,名称以Parallel开头的回收器,多线程回收,全程stw: 并发回收,cms与G1,多线程分阶段回收,只有某阶段会stw: CMS垃圾回收 CMS垃圾回收特点 cms只会回收老年代和永久带(1.8开始为元数据区,需要设置CMSClassUnloadingEnabled),不会收

JVM的判断对象是否已死和四种垃圾回收算法总结

面试题一:判断对象是否已死 判断对象是否已死就是找出哪些对象是已经死掉的,以后不会再用到的,就像地上有废纸.饮料瓶和百元大钞,扫地前要先判断出地上废纸和饮料瓶是垃圾,百元大钞不是垃圾.判断对象是否已死有引用计数算法和可达性分析算法. 1.引用计数算法 给每一个对象添加一个引用计数器,每当有一个地方引用它时,计数器值加 1:每当有一个地方不再引用它时,计数器值减 1,这样只要计数器的值不为 0,就说明还有地方引用它,它就不是无用的对象.如下图,对象 2 有 1 个引用,它的引用计数器值为 1,对象

【转】V8 之旅: 垃圾回收器

垃圾回收器是一把十足的双刃剑.其好处是可以大幅简化程序的内存管理代码,因为内存管理无需程序员来操作,由此也减少了(但没有根除)长时间运转的程序的内存泄漏.对于某些程序员来说,它甚至能够提升代码的性能. 另一方面,选择垃圾回收器也就意味着程序当中无法完全掌控内存,而这正是移动终端开发的症结.对于JavaScript,程序中没有任何内存管理的可能--ECMAScript标准中没有暴露任何垃圾回收器的接口.网页应用既没有办法管理内存,也没办法给垃圾回收器进行提示. 严格来讲,使用垃圾回收器的语言在性能

深入理解JVM虚拟机3:垃圾回收器详解

JVM GC基本原理与GC算法 微信公众号[Java技术江湖]一位阿里 Java 工程师的技术小站.作者黄小斜,专注 Java 相关技术:SSM.SpringBoot.MySQL.分布式.中间件.集群.Linux.网络.多线程,偶尔讲点Docker.ELK,同时也分享技术干货和学习经验,致力于Java全栈开发!(关注公众号后回复”Java“即可领取 Java基础.进阶.项目和架构师等免费学习资料,更有数据库.分布式.微服务等热门技术学习视频,内容丰富,兼顾原理和实践,另外也将赠送作者原创的Jav

垃圾回收器如何确定哪些对象要回收---《深入理解java虚拟机》

垃圾回收器如何确定哪些对象要回收: 引用计数法 很多教科书判断对象是否存活的算法是这样的:给对象添加一个引用计数器,每当有一个地发引用它时,计数器值就加1:当引用失效时,计数器值就减1:任何时刻计数器都为0的对象就是不可能再被使用的. 客观地说,引用技术算法的实现简单,判定效率也很高,在大部分情况下它都是一个不错的算法,也有一些比较著名的应用案例,例如微软的COM技术.使用ActionScript 3 的FlashPlayer.Python语言以及在游戏脚本领域中被广泛应用的Squirrel中都

谈一谈垃圾回收器

目的: 使用垃圾回收器的唯一原因就是:回收程序不再使用的内存. 针对的目标对象: Java的垃圾回收器会自动回收不再使用的Java对象,释放内存.但是回收的是用new创建的,分配在堆上的内存. finalize(): 那么,如果不是用这种方式创建的对象,该怎么回收?比如:Java调用了本地的c语言方法创建了个对象,那么这时,该对象不是放在堆上的.除非你手动去调用c的free()方法,否则,这个对象将永远不会被清理. Java的finalize()方法可以解决上面的问题.垃圾回收器在回收垃圾对象时