文章目录
一、字符串常量池
JDK1.8
之前,Hotspot
虚拟机中,字符串常量池存放于方法区的实现永久代(Perm Space)中,JDK1.8
开始,字符串常量池移到了堆中。
字符常量池本质上是一张哈希表,key
为由字符串内容及其长度生成的哈希值,value
为字符串对象在堆中的地址。
二、常量池与运行时常量池
public class T01_ConstantPool {
public static void main(String[] args) {
}
}
运行上面的程序后,可在项目目录下找到
T01_ConstatntPool.class
文件,我们将目录切换到class文件所在目录,运行命令javap -v T01_ConstantPool.class
,在运行结果中可以看到类似下方的Constant pool
信息。
Constant pool:
#1 = Methodref #3.#17 // java/lang/Object."<init>":()V
#2 = Class #18 // T01_Pool
#3 = Class #19 // java/lang/Object
#4 = Utf8 <init>
#5 = Utf8 ()V
#6 = Utf8 Code
#7 = Utf8 LineNumberTable
#8 = Utf8 LocalVariableTable
#9 = Utf8 this
#10 = Utf8 T01_ConstantPool;
#11 = Utf8 main
#12 = Utf8 ([Ljava/lang/String;)V
#13 = Utf8 args
#14 = Utf8 [Ljava/lang/String;
#15 = Utf8 SourceFile
#16 = Utf8 T01_ConstantPool.java
#17 = NameAndType #4:#5 // "<init>":()V
#18 = Utf8 T01_Pool
#19 = Utf8 java/lang/Object
这个
Constant pool
就是常量池,编译时每个类都会产生类似的产量池,我们可以看到,每个常量指向的都是一个符号引用(类/方法/属性的全限定名)。
到了运行时期,Constant pool
常量池中符号引用会被转化为直接引用(类/方法/属性在内存中的真实地址),从而Constant pool
常量池就转变为了运行时常量池。运行时常量池位于方法区(永久代/元空间)。