深入理解JVM——字符串常量池

【问题】:请判断以下打印结果:

public class RuntimeConstantPoolOOM {
		public static void main(String[] agrs) {
			String str1 = new StringBuilder("计算机").append("软件").toString();
			System.out.println(str1.intern() == str1);
			
			String str2 = new StringBuilder("ja").append("va").toString();
			System.out.println(str2.intern() == str2);
		}
	}	

上述代码具体打印结果在不同JDK版本中并不相同。其原因是在JDK1.7版本中,原存储在方法区(永久代)中的字符串常量池被移到了堆(Heap)空间中。因此导致上述代码打印结果的不同。

JDK1.6及之前

【答案】:两个都是(false false)

false
false

【原因】:在JDK1.6及以前的版本中,字符串常量池存储在方法区(永久代)中,intern() 方法会把首次出现的字符串实例复制到字符串常量池中(永久代),并返回该常量池中字符串实例的引用。而由StringBulider创建的str1对象则是存放在堆(Heap)中,因此两者的引用必然不同。

JDK1.7及之后

【答案】:一对一错(true false)

true
false

【原因】:在JDK1.7版本中,字符串常量池和静态变量都被移到了java堆空间中,此时的 intern() 方法不在需要将首次遇到的字符串实例负责到永久代中(此时字符串常量池已在堆中),只需要记录首次出现实例的引用即可,因此与 StringBulider 创建的实例对象是同一引用,结果为 true。

但是为什么第二个答案却是 false 呢?这是因为 “java” 字符串并不是StringBulider 创建时首次出现,不符合 intern() 首次遇见原则,两者自然不少同一引用。
注:"java"字符串是在加载 sun.misc.Version 这个类时被加载到字符串常量池中的。

结论

要清楚在JDK1.7版本中,将字符串常量池和静态变量从方法区(永久代)移动到了java堆中,这导致intern() 方法的实现出现了变化。此外,在JDK1.8中,永久代(PermGen Space)被移除,取而代之的是采用本地内存元空间(MetaSpace),而字符串常量池和静态变量仍然存放在java堆中。理解这些内容之后,再遇到字符串常量池相关问题就能做到迎刃而解。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 字符串常量JVM中属于方法区(也称为永久代)内存分区。方法区是各个线程共享的内存区域,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。字符串常量是方法区中的一部分,用于存储字符串常量。在JDK 8之后,方法区被取消了,取而代之的是元空间(MetaSpace),但字符串常量仍然存放在元空间中。 ### 回答2: 字符串常量JVM的方法区(也称为非堆区)中。 JVM将内存分为几个不同的区域,包括堆区、方法区、虚拟机栈等。而字符串常量是方法区的一部分,用于存储在程序中直接使用的字符串常量。 在Java中,字符串常量是一种特殊的内存存储区域,用于存储字符串常量,它的作用是提高字符串的重用性和效率。当我们使用双引号声明一个字符串时,JVM会首先在字符串常量中查找是否存在相同内容的字符串,如果存在则直接返回引用,如果不存在则创建一个新的字符串并放入字符串常量中。这种机制可以减少内存占用,提高程序的执行效率。 由于字符串常量位于方法区,它是与其他线程共享的,在程序运行过程中,多个线程可以同时访问字符串常量。而且,字符串常量的位置是在程序的执行过程中被动态调整的,当字符串没有被引用时,JVM会自动回收字符串常量中的空间。 总结来说,字符串常量JVM的方法区的一部分,用于存储程序中直接使用的字符串常量,并提高字符串的重用性和效率。 ### 回答3: 字符串常量JVM的方法区里。方法区是JVM的一个内存分区,用于存储类信息、常量、静态变量、即时编译器编译后的代码等。而字符串常量就是方法区的一部分,用于存储字符串常量。 在Java中,当我们使用字符串字面量(如"hello")时,编译器会将其放入字符串常量中。当程序执行时,如果再次使用相同的字符串字面量,JVM会直接从字符串常量中取出已存在的字符串对象,而不会创建新的对象,这样可以节省内存空间。 由于字符串在Java中使用非常频繁,所以将字符串常量放在方法区中,可以提高字符串的重用率。此外,字符串常量的位置在方法区中也有利于GC(垃圾回收),因为当某个字符串不再被引用时,GC可以更方便地回收该字符串常量。 需要注意的是,从Java 7开始,字符串常量被移出了PermGen空间(方法区的前身),并放置在堆中,这是因为字符串常量中的字符串对象是可以被垃圾回收的,而且过多的字符串常量可能导致PermGen空间溢出的问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值