首先说明,此方法区的演进指的是HotSpot虚拟机
jdk1.6及之前 | 有永久代(permanent generation) ,静态变量存放在 永久代,上 |
jdk1.7 | 有永久代,但已经逐步“去永久代”,字符串常量池、静 态变量移除,保存在堆中 |
jdk1.8及之后 | 无永久代,类型信息、字段、方法、常量保存在本地内存 的元空间,但字符串常量池、静态变量仍在堆 |
JKD1.6时方法区内存储的是:类型信息、运行时常量池、 静态变量、 JIT代码缓存、 域信息、 方法信息
类型信息:
这个类型的完整有效名称(全名=包名.类名)
这个类型直接父类的完整有效名(对于interface或是java. lang . object, 都没有父类)
这个类型的修饰符(public, abstract, final的某个子集)
这个类型直接接口的一个有序列表
域信息:
域名称、域类型、域修饰符(public, private,protected, static, final, volatile, transient的某个子集)
方法信息
JVM必须保存所有方法的以下信息,同域信息一样包括声明顺序:
方法名称
方法的返回类型(或void)
方法参数的数量和类型(按顺序)I
方法的修饰符(public,private, protected, static, final,synchronized, native,abstract的一个子集)
方法的字节码(bytecodes)、操作数栈、局部变量表及大小( abstract和native方法除外)
异常表( abstract和native方法除外)
每个异常处理的开始位置、结束位置、代码处理在程序计数器中的偏移地址、被捕获的异常类的常量池索引
运行时常量池
● 运行时常量池( Runtime Constant Pool)是方法区的一部分。
●常量池表(Constant Pool Table) 是Class文件的一部分,用于存放编译期生成的各
种字面量与符号引用,这部分内容将在类加载后存放到方法区的运行时常量池中。
●运行时常量池,在加载类和接口到虚拟机后,就会创建对应的运行时常量池。
●JVM为每个已加载的类型(类或接口)都维护一个常量池。池中的数据项像数组项- -样,
是通过索引访问的。
●运行时常量池中包含多种不同的常量,包括编译期就已经明确的数值字面量,也包括到
运行期解析后才能够获得的方法或者字段引用。此时不再是常量池中的符号地址了,这
里换为真实地址。
➢运行时常量池,相对于Class文件常量池的另一-重要特征是:具备动态性。
●运行时常量池类似于传统编程语言中的符号表(symbol table) ,但是它所包含的数
据却比符号表要更加丰富-一些。
●当创建类或接口的运行时常量池时,如果构造运行时常量池所需的内存空间超过了方法
区所能提供的最大值,则JVM会抛OutOfMemoryError异常。
JKD1.6:
JKD1.7:
JDK1.8: