我终于搞清楚了和String有关的那点事儿

作者:网友投稿 时间:2018-06-27 16:56

字号
技术沙龙 | 6月30日与多位专家探讨技术高速发展下如何应对运维新挑战!

String,是Java中除了基本数据类型以外,最为重要的一个类型了。很多人会认为他比较简单。但是和String有关的面试题有很多,下面我随便找两道面试题,看看你能不能都答对:

Q1:String s = new String("hollis");定义了几个对象。

Q2:如何理解String的intern方法?

上面这两个是面试题和String相关的比较常考的,很多人一般都知道答案。

A1:若常量池中已经存在"hollis",则直接引用,也就是此时只会创建一个对象,如果常量池中不存在"hollis",则先创建后引用,也就是有两个。

A2:当一个String实例调用intern()方法时,JVM会查找常量池中是否有相同Unicode的字符串常量,如果有,则返回其的引用,如果没有,则在常量池中增加一个Unicode等于str的字符串并返回它的引用;

两个答案看上去没有任何问题,但是,仔细想想好像哪里不对呀。

我终于搞清楚了和String有关的那点事儿

按照上面的两个面试题的回答,就是说new String会检查常量池,如果有的话就直接引用,如果不存在就要在常量池创建一个,那么还要intern干啥?难道以下代码是没有意义的吗?

String s = new String("Hollis").intern(); 

如果,每当我们使用new创建字符串的时候,都会到字符串池检查,然后返回。那么以下代码也应该输出结果都是true?

String s1 = "Hollis"

    String s2 = new String("Hollis"); 

    String s3 = new String("Hollis").intern(); 

 

    System.out.println(s1 == s2); 

    System.out.println(s1 == s3); 

但是,以上代码输出结果为(base jdk1.8.0_73):

false 

true 

不知道,聪明的读者看完这段代码之后,是不是有点被搞蒙了,到底是怎么回事儿?

别急,且听我慢慢道来。

字面量和运行时常量池

JVM为了提高性能和减少内存开销,在实例化字符串常量的时候进行了一些优化。为了减少在JVM中创建的字符串的数量,字符串类维护了一个字符串常量池。

在JVM运行时区域的方法区中,有一块区域是运行时常量池,主要用来存储编译期生成的各种字面量和符号引用。

了解Class文件结构或者做过Java代码的反编译的朋友可能都知道,在java代码被javac编译之后,文件结构中是包含一部分Constant pool的。比如以下代码:

public static void main(String[] args) { 

    String s = "Hollis"

经过编译后,常量池内容如下:

Constant pool: 

  #1 = Methodref          #4.#20         // java/lang/Object."<init>":()V 

  #2 = String             #21            // Hollis 

  #3 = Class              #22            // StringDemo 

  #4 = Class              #23            // java/lang/Object 

  ... 

  #16 = Utf8               s 

  .. 

  #21 = Utf8               Hollis 

  #22 = Utf8               StringDemo 

  #23 = Utf8               java/lang/Object 

上面的Class文件中的常量池中,比较重要的几个内容:

#16 = Utf8               s 

  #21 = Utf8               Hollis 

  #22 = Utf8               StringDemo 

责任编辑:CQITer新闻报料:400-888-8888   本站原创,未经授权不得转载
关键词 >>String Java 数据
继续阅读
热新闻
推荐
关于我们联系我们免责声明隐私政策 友情链接