方法区特点
-
保存在着被加载过的每一个类的信息;这些信息由类加载器在加载类的时候,从类的源文件中抽取出来;static变量信息也保存在方法区中;
-
可以看做是将类(Class)的元数据,保存在方法区里;
-
方法区是线程共享的;当有多个线程都用到一个类的时候,而这个类还未被加载,则应该只有一个线程去加载类,让其他线程等待;
-
方法区的大小不必是固定的,jvm可以根据应用的需要动态调整。jvm也可以允许用户和程序指定方法区的初始大小,最小和最大限制;
-
方法区同样存在垃圾收集,因为通过用户定义的类加载器可以动态扩展Java程序,这样可能会导致一些类,不再被使用,变为垃圾。这时候需要进行垃圾清理。
在jdk1.8之前,是用永久代来实现了方法区
在jdk1.8以后使用的是本地内存的元空间实现,不再占用堆内存
内存溢出问题
- 1.8以前会导致永久代内存溢出
- 1.8以后会导致元空间内存溢出
//修改提供的元空间大小
//-XX:MaxMetaspaceSize=8m
//修改永久代大小
//-XX:MaxPermSize=8m
class ClassHeap extends ClassLoader{
public static void main(String[] args){
int j=0;
try {
ClassHeap test=new ClassHeap();
for(int i=0;i<10000;i++,j++){
//ClassWriter 作用是生成类的二进制字节码
ClassWriter cw=new ClassWriter(0);
//版本号,public,类名,包名,父类,接口
cw.visit(Opcodes.V1_8,Opcodes.ACC_PUBLIC,"Class"+1,null,"java/lang/Object",null);
//返回BYTE
byte[] code=cw.toByteArray();
//执行了类的加载
test.defineClass("Class"+1,code,0,code.length);
}
}finally {
System.out.println(j);
}
}
运行时常量池
- 常量池,就是一张表,虚拟机指令根据这张常量表找到要执行的类名,方法名,参数变量,字面量等信息
- 运行时常量池,常量池是*.class文件中的,当该类被加载,它的常量池信息会放入运行时常量池,并把里面的符号地址变为真实地址