方法区(Method Area)与Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类型信息、常量、静态变量、即时编译器编译后的代码缓存等数据。(如图所示,出自《深入理解Java虚拟机》)
方法区具体包含内容为:
- 类型信息
对每个加载的类型(类class、接口interface、枚举enum、注解annotation),JVM方法区中存储以下类型信息:这个类型的完整有效名称(全名=包名.类名)、这个类型直接父类的完整有效名(对于interface或是java.lang.Object,都没有父类)、这个类型的修饰(public,abstract,final的某个子集)、这个类型直接接口的一个有序列表。
域(Field)信息
JVM必须在方法区中保存类型的所有域的相关信息以及域的声明顺序。域的相关信息包括:域名称、域类型、域修饰符(public,private,protected,static,final,volatile,transient的某个子集)
方法(Method)信息JVM必须保存所有方法的以下信息,同域信息一样包括声明顺序:
方法名称、方法的返回类型、方法参数的数量和类型(按顺序)、方法的修饰符(public,private,protected,static,final,synchronized,native,abstract的一个子集)、方法的字节码(bytecodes)、操作数栈、局部变量表及大小(abstract和native方法除外)、异常表(abstract和native方法除外)、每个异常处理的开始位置、结束位置、代码处理在程序计数器中的偏移地址、被捕获的异常类的常量池索引。
non-final的类变量
静态变量和类关联在一起,随着类的加载而加载,他们成为类数据在逻辑上的一部分。类变量被类的所有实例共享,即时没有类实例时你也可以访问它。
全局常量:static final被声明为final的类变量的处理方法则不同,每个全局常量在编译的时候就会被分配了。
方法区溢出
不断地生成CLASS类导致方法区溢出
为了加快方法区溢出的效果,需要在下方增加如下参数,限制栈的大小。
-XX:+PrintGCDetails
-XX:+PrintGCDateStamps
-XX:-UseCompressedClassPointers
-XX:MetaspaceSize=20M
-XX:MaxMetaspaceSize=20m
第一个参数用于打印GC日志;
第二个参数用于打印对应的时间戳;
第三个参数-XX:-UseCompressedClassPointer表示在Metaspace中不要开辟出一块新的空间(Compressed Class Space),如果开辟这块空间的话,该空间默认大小是1G,所以我们关闭该功能,此时再设置Metaspace的大小;
结果: