1.概述
方法区的回收,看起来是存在一定争议的。JVM虚拟机规范中并没有要求虚拟机强制回收方法区。在书中,也讲解了,JDK11版本的ZGC,其实就不包含方法区回收的内容。因为,相对堆而言,方法区回收内容比较少。其实常见的回收只有运行时常量池回收以及类卸载。
2.方法区回收
2.1 常量池回收
这里的常量池回收其实回收的是废弃常量,就像String的intern()方法能够往运行时常量池添加数据一样,当这个常量池中的字符串在程序中一点都没有用到的时候,就可以被回收了。
2.2 类卸载
类的回收,相对而言比较困难。首先,类信息要回收,必须得保证其他地没有用到这个类的地方,必要的条件三个:
在当前进程中,所有根据这个类创建的对象必须都被回收。
在当前进程中,这个类的所有类加载器也被回收。
在当前进程中,不存在引用了这个类的Class对象,就是反射的这套。
在满足这三个条件的情况下,类才有被回收的资格。如果想被回收,需要一些JVM参数配合:
-Xnoclassgc 是否对类型进行回收
-verbose:class
-XX:TraceClassLoading: 查看类加载信息
-XX:TraceClassUnLoading: 查看类卸载信息
场景的话,在大批量使用反射,动态代理,动态生成JSP等,需要具备类卸载的能力,防止系统造成太大压力。
感觉,正常用不到啊,这个考虑一下与后续的关联性吧。