在《java虚拟机规范》中,并没有强制要求堆方法区实现垃圾回收,不同落地实现的虚拟机可以选择实现或者不实现方法区的垃圾回收,要是不实现垃圾回收的话,那么方法区就成真正意义上的‘永久代’了,对于Hotspot虚拟机来说,它是实现了方法区的垃圾回收的
一般来说方法区的回收效果比较令人满意,尤其是类型的卸载,条件非常苛刻,但是是必要的,方法区的垃圾收集主要分为两部分内容:
- 常量池中废弃的常量
- 不再使用的类型
首先我们需要知道,方法区内常量池中存放的两大类常量,字面量和符号引用,字面量比较接近于java语言层次的常量概念,如文本字符串,被声明为final
的常量值等,而符号引用包括下面三类常量:
- 1.类和接口的全限定名
- 2.字段的名称和描述符
- 3.方法的名称和描述符
HotSpot虚拟机对常量池的回收策略是很明确的,只要常量池中的常量没有被任何地方引用,就可以被回收 那么该如何判断一个类型是否废弃呢?
与常量的回收不同,类型是否被废弃需要同时满足下面三个条件:
- 该类所有的实例都已被回收,也就是java堆中不存在该类及其任何派生子类的实例
- 加载该类的类加载器已被回收
- 该类对应的
java.lang.Class
对象没有在任何地方被引用,无法在任何地方通过反射访问该类的方法
Java虚拟机允许同时满足以上三个条件的类进行回收,注意这里是‘允许’,不一定会回收,和对象的没有引用就必然会被回收不同