深入了解Java字符串常量池

String s = new String("远飞的梦");

面试题1:这行代码创建了几个对象 ?

答:创建了两个对象。使用 new 关键字创建一个字符串对象时,Java 虚拟机会先在字符串常量池中查找有没有‘远飞的梦’这个字符串对象,如果有就不会在字符串常量池中创建‘远飞的梦’这个对象了,它会直接在堆中创建一个‘远飞的梦’的字符串对象,然后将堆中这个"远飞的梦"的对象地址返回赋值给变量 s。

疑问:为什么要先在字符串常量池中创建对象,然后再在堆上创建呢?这样不就多此一举了?

解答:由于字符串的使用频率实在是太高了,所以 Java 虚拟机为了提高性能和减少内存开销,在创建字符串对象的时候进行了一些优化,特意为字符串开辟了一个字符串常量池。

通常情况下,我们会采用双引号的方式来创建字符串对象,而不是通过 new 关键字的方式:

String s = "今天很开心";

当执行 String s = "今天很开心" 时,Java 虚拟机会先在字符串常量池中查找有没有“今天很开心”这个字符串对象,如果有,则不创建任何对象,直接将字符串常量池中这个“三妹”的对象地址返回,赋给变量 s;如果没有,在字符串常量池中创建“三妹”这个对象,然后将其地址返回,赋给变量 s。有了字符串常量池,就可以通过双引号的方式直接创建字符串对象,不用再通过 new 的方式在堆中创建对象了。ew 的方式始终会创建一个对象,不管字符串的内容是否已经存在,而双引号的方式会重复利用字符串常量池中已经存在的对象。

疑问:字符串常量池在内存中的什么位置呢?

答:在 Java 8 之前,字符串常量池在永久代中。

       Java 8 之后,移除了永久代,字符串常量池就移到了堆中。

问题:方法区,永久代和元空间分别是什么?

  • 方法区是 Java 虚拟机规范中的一个概念,就像是一个接口吧;
  • 永久代是 HotSpot 虚拟机中对方法区的一个实现,就像是接口的实现类;
  • Java 8 的时候,移除了永久代,取而代之的是元空间,是方法区的另外一个实现。

永久代是放在运行时数据区中的,所以它的大小受到 Java 虚拟机本身大小的限制,所以 Java 8 之前,会经常遇到 java.lang.OutOfMemoryError: PremGen Space 的异常,PremGen Space 就是方法区的意思;而元空间是直接放在内存中的,所以只受本机可用内存的限制,虽然也会发生内存溢出,但出现的几率相对之前就小了很多。

你可能感兴趣的