一般来说方法区的回收很难令人满意,因为条件苛刻所以回收效果不太好,但又是需要的。方法区主要回收的是两个东西:类型信息和运行时常量池。
运行时常量池
字面量和符号引用,字面量主要包括三类:
- 类和接口的全限定名
- 字段的名称和描述符
- 方法的名称和描述符
只要常量池的常量没有被引用,就可以回收。
类型信息
判断一个类型不再被使用的条件比较苛刻,需要满足以下三个条件:
- 该类的所有实例都被回收了
- 加载该类的类加载器已经被回收,除非是经过精心设计的可替换类加载器的场景,如OSGi、JSP的重加载等,否则是很难达成的
- 该类对应的java.lang.Class对象没有在任何地方被引用,也没有任何地方通过反射访问该类的方法。
满足了以上条件后,就允许被回收,但不一定能被回收,hotspot提供了-Xnoclassgc参数进行控制是否要对类型进行回收。
在大量使用反射、动态代理、CGLib等字节码框架,频繁自定义类加载器的场景中,就需要jvm具备类型卸载的能力,保证不会对对方法区造成过大的内存压力。