一、对象是否存活
1、判断方法
1)引用计数算法:给对象添加一个引用计数器,每当有一个地方引用它时,计数器值加1;当引用失效时,计数器值减1;任何时刻计数器为0的对象就是不可能再被使用的。
存在问题:很难解决对象间相互循环引用的问题
2)可达性分析算法:通过一系列的称为“GC Roots”的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到GC Roots没有任何引用链时(从GC Roots到这个对象不可达),则证明此对象是不可用的。
可作为GC Roots的对象:虚拟机栈(栈帧中的本地变量表)中引用的对象;
方法区中类静态属性引用的对象;
方法区中常量引用的对象;
本地方法栈中JNI(即一般说的Native方法)引用的对象
2、标记一个对象死亡,至少需要经历两次标记过程:
第一次标记与筛选:对象在进行可达性分析后发相没有与GC Roots 相连接的引用链。筛选的条件是此对象是否有必要执行finalize()方法
----当对象没有覆盖finalize()方法或者已经被虚拟机调用过,则虚拟机将这两种情况视为“没有必要执行”
----当对象被判定为有必要执行finalize()方法,则对象被放置在F-Queue的队列中,由finalizer线程去执行。
第二次标记:GC会对F-Queue中的对象进行第二次小规模标记。若对象建立起关联则被移除“即将回收”的集合,否则回收。
3、关于回收方法区
永久代垃圾回收内容:废弃常量(必然回收)和无用的类(可回收也可不回收)
----废弃常量:没有被引用的常量
----无用的类判断条件:类中所有实例都已经被回收;加载该类的ClassLoader已经被回收;该类对应的java.lang.Class对象没有在任何地方被引用,无法通过反射访问该类方法。
通过-Xnoclassgc参数控制是否对类进行回收;
通过-verbose:class以及-XX:+TraceClassLoading(Product版虚拟机)
-XX:+TraceClassUnloading(FastDebug版虚拟机)查看类加载卸载信息
二、引用分类:强引用,软引用,弱引用,虚引用
- 强引用(Strong Reference):指在程序代码中普遍存在的,类似“Object obj =new Object()”之类的引用。强引用存在时,垃圾收集器永远不会回收掉被引用的对象
- 软引用(Soft Renference):描述一些还有用但并非必须的对象,在内存将要溢出时进行回收,若回收后内存仍不够,才抛出内存溢出异常
- 弱引用(WeakReference):强度比软引用更弱。垃圾收集器工作时,无论内存是否足够都会回收
- 虚引用(Phantom Reference):最弱的引用关系。为一个对象设置虚引用关联的唯一目的就是能在这个对象被收集器回收时收到一个系统通知