回收方法区
回收的必要性
Java 虚拟机规范中,可以不实现垃圾回收。方法区(HotSpot虚拟机中的永久代)进行垃圾回收时相对于堆中垃圾收回而言,“性价比”比较低。
方法区中的存储
方法区中存储的是加载的类的信息,常量,静态变量,即时编译后的代码等数据,所以回收的对象也就是这些内容。
何种场景需要回收
既然回收方法区不是必须的,虽然效率低下,但是当内存不够使用的时候依然是会抛出OOM的,那么我们需要知道什么场景下需要去回收方法区。首先我们需要弄明白方法区会回收那些对象。在JDK1.7之前,常量池是在方法区中的,在此版本及以后则将其移到堆中。基于目前版本主要是1.8及以上,故我们以1.8为准。在此版本上,主要回收的是无用的类。如何判定一个类是无用的:
1.该类的所有实例都已经被回收;
2.加载该类的ClassLoader已经被回收;
3.该类对应的java.lang.Class对象没有在被任何地方被引用,无法在任何地方通过反射访问该类的方法。
因此,在大量使用反射,动态代理,CGlib等ByteCode框架,动态生成JSP以及OSGi这类频繁自定义ClassLoader的场景都需要虚拟机具备类卸载的功能,以保证永久代也就是方法区内存不会溢出。
如何配置
既然知道什么情况下可以回收无用的类,接下来我们看看怎么配置使我们的虚拟机具备回收无用类的功能。HotSpot配置项:
-Xnoclassgc表示不进行回收;
Product版本中,可以使用-verbose:class和-XX:+TraceClassLoading查看类的加载信息
FastDebug版本中,可以使用-XX:+TraceClassUnloading查看类的卸载信息