java虚拟机 内存模型之常量池

详见我的另一篇文章:JVM内存模型《JVM内存模型》
java虚拟机 内存模型之常量池

  1. 程序计数器PCR是jvm执行程序的流水线,存放一些跳转指令。
  2. 本地方法栈是jvm调用操作系统方法所使用的栈。
  3. 虚拟机栈是jvm执行java代码所使用的栈(局部变量表(栈)、操作数栈(数值计算)、动态链接、附加信息、返回地址(帮助栈帧去恢复上层方法的状态))。
  4. 方法区存放了一些常量、静态变量、类信息等,可以理解成class文件在内存中的存放位置(运行时常量池)。
  5. 虚拟机堆是jvm执行java代码所使用的堆。

(静态)常量池中主要保存的是一些字面变量和符号引用。

字面变量:可以理解为java语言中的常量;如文本字符串,声明为final的常量值等,符号
符号引用:包括类和接口的全限定名、方法和字段的名称和描述符。

常量池主要分为静态常量池和运行时常量池。
区别显而易见,后者可以在非编译期(运行时)将常量放入池中(具体解释见下文)。

  1. 静态常量池又叫class文件中 的常量池,顾名思义,就是存放.class文件中的常量信息,不仅仅包含字符串(数字)字面量,还包含类、方法的信息,占用class文件绝大部分空间。这种常量池主要用于存放两大类常量:字面量(Literal)和符号引用量(Symbolic References)
  2. 运行常量池是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

java虚拟机 内存模型之常量池
java虚拟机 内存模型之常量池

结论

必须要关注编译期的行为,才能更好的理解常量池。
运行时常量池中的常量,基本来源于各个class文件中的常量池。
程序运行时,除非手动向常量池中添加常量(比如调用intern方法),否则jvm不会自动添加常量到常量池。

部分转自:https://www.cnblogs.com/syp172654682/p/8082625.html