Java虚拟机 String常量池浅析

前言

我们来剖析一下关于字符串常量池。废话不多说,直接进入正题。

@Test
public void testString(){
    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 = "hel";
    String s8 = "lo";
    String s9 = s7 + s8;
    System.out.println(s1 == s2);
    System.out.println(s1 == s3);
    System.out.println(s1 == s4);
    System.out.println(s4 == s5);
    System.out.println(s5 == s6); //false
    System.out.println(s1 == s9);
    System.out.println(s1 == s6); //ture
}

我们一个一个来解析:

  • s1==s2为true:

因为s1,s2赋值的时候均使用的是字符串字面量。在编译期间,这种字符串字面量会直接进入Class文件的常量池中,在运行的时候会进入方法区的运行时常量池。所以,s1,s2指向的都是同一片内存地址。

s1和s2
  • s1==s3为true:

s3虽然是动态拼接出来的字符串,但是所有参与拼接的部分都是已知的字符串字面量,所以JVM会对其进行优化,s3="hel" + "lo"会在编译期直接被优化成s3=“hello”,其余部分则与第一种情况相同,不再赘述。

  • s1==s4为false:

s4虽然也是动态拼接出来的字符串,但是new String("lo")不是已知的字面量,是一个不可预料的值,编译器不会优化,必须等到运行时才能知道结果。

我们可以看到s1和s4所指的对象是不同的,一个在方法区中,一个在Java堆中,所以判断s1==s4的答案为false。

  • s4==s5答案为false:这个就不需要解释了,两个对象都在Java堆中,但是内存地址不同,所以不相等。
  • s1==s9为false:这个也不需要解释,和第三种情况雷同。
  • s5==s6为false,而s1==s6为ture:这个着重分析一下:

这里主要是intern的作用,基于JDK1.8(不同版本可能结果不同)。
首先s5是存在Java堆中的对象,使用intern方法会尝试将字符串字面量“hello”加入到运行时常量池中,如果此时常量池池没有该字面量,则会创建一个并返回其在常量池中的地址,但是此时常量池已经有"hello"这个字面量了,那么直接返回其在常量池的地址。因此,s1与s6指向的都是同一个地址,而s5和s6指向的则不是同一个内存地址,一个在Java堆中,一个在运行时常量池中。


至此,关于字符串常量池的一些区别,我们已经全部分析完。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 从网上复制的,看别人的比较全面,自己搬过来,方便以后查找。原链接:https://www.cnblogs.com/...
    lxtyp阅读 1,360评论 0 9
  • 一、概述 常量池:编译期被确定,*.class文件中的一部分,包含字面量(Literal)和符号引用(Symbol...
    高稷阅读 1,933评论 0 2
  •   需要说明的一点是,这篇文章是以《深入理解Java虚拟机》第二版这本书为基础的,这里假设大家已经了解了JVM的运...
    Geeks_Liu阅读 14,091评论 5 44
  • 本文是我自己在秋招复习时的读书笔记,整理的知识点,也是为了防止忘记,尊重劳动成果,转载注明出处哦!如果你也喜欢,那...
    波波波先森阅读 865评论 1 6
  • package cn.itcast_01;/* 字符串:就是由多个字符组成的一串数据。也可以看成是一个字符数组。 ...
    蛋炒饭_By阅读 622评论 0 0