详见我的另一篇文章:JVM内存模型《JVM内存模型》
- 程序计数器PCR是jvm执行程序的流水线,存放一些跳转指令。
- 本地方法栈是jvm调用操作系统方法所使用的栈。
- 虚拟机栈是jvm执行java代码所使用的栈(局部变量表(栈)、操作数栈(数值计算)、动态链接、附加信息、返回地址(帮助栈帧去恢复上层方法的状态))。
- 方法区存放了一些常量、静态变量、类信息等,可以理解成class文件在内存中的存放位置(运行时常量池)。
- 虚拟机堆是jvm执行java代码所使用的堆。
声明一下:java8中的方法区(永久代)已经被取消,改用元空间代替(Metaspace),JVM把String常量池移入了堆中。
(静态)常量池中主要保存的是一些字面变量和符号引用。
字面变量:可以理解为java语言中的常量;如文本字符串,声明为final的常量值等,符号
符号引用:包括类和接口的全限定名、方法和字段的名称和描述符。
常量池主要分为静态常量池和运行时常量池。
区别显而易见,后者可以在非编译期(运行时)将常量放入池中(具体解释见下文)。
- 静态常量池又叫class文件中 的常量池,顾名思义,就是存放.class文件中的常量信息,不仅仅包含字符串(数字)字面量,还包含类、方法的信息,占用class文件绝大部分空间。这种常量池主要用于存放两大类常量:字面量(Literal)和符号引用量(Symbolic References)。
- 运行常量池是JVM内存模型中方法区的一部分。主要是在jvm虚拟机完成类加载的操作之后,将class文件中的常量池加载到内存中,保存在方法区中。这也是我们常说的常量池。
运行时常量池相对于Class文件常量池的另外一个重要特征是具备动态性,Java语言并不要求常量一定只有编译期才能产生,也就是并非预置入Class文件中常量池的内容才能进入方法区运行时常量池,运行期间也可能将新的常量放入池中,这种特性被开发人员利用比较多的就是String类的intern()方法。
最常见的就是String的intern():String的intern()方法会查找在常量池中是否存在一份equal相等的字符串,如果有则返回该字符串的引用,如果没有则添加自己的字符串进入常量池。
常量池的好处
常量池是为了避免频繁的创建和销毁对象而影响系统性能,其实现了对象的共享。
例如字符串常量池,在编译阶段就把所有的字符串文字放到一个常量池中。
(1)节省内存空间:常量池中所有相同的字符串常量被合并,只占用一个空间。
(2)节省运行时间:比较字符串时,= = 比equals()快。对于两个引用变量,只用==判断引用是否相等,也就可以判断实际值是否相等。
贴上常见的代码示例:
String s1 = "Hello";
String s2 = "Hello";
String s3 = "Hel" + "lo";
String s4 = "Hel" + new String("lo");
String s5 = new String("Hello");
String s6 = s5.intern();
String s7 = "H";
String s8 = "ello";
String s9 = s7 + s8;
System.out.println(s1 == s2); // true
System.out.println(s1 == s3); // true
System.out.println(s1 == s4); // false
System.out.println(s1 == s9); // false
System.out.println(s4 == s5); // false
System.out.println(s1 == s6); // true
结论
必须要关注编译期的行为,才能更好的理解常量池。
运行时常量池中的常量,基本来源于各个class文件中的常量池。
程序运行时,除非手动向常量池中添加常量(比如调用intern方法),否则jvm不会自动添加常量到常量池。