Java Stirng.intern 和 StringBuilder 的疑惑

在学习JVM内存的时候遇到了一个问题。我的环境是jdk1.8, 本来想通过intern验证Java8的
内存中字符常量是存在heap里面的,intern不会复制第一次遇到的字符串,而是把引用放入
字符串常量池。

但是在下面代码里:
String a = new StringBuilder("计算机").toString();

        System.out.println(a.intern()==a);

输出的是false

而在:
String a = new StringBuilder("计算机").append("aaa").toString();

    System.out.println(a.intern()==a);

中输出了true。

我的理解是都输出true。不知道为什么第一次会输出false?

4个回答

我看到你发的另外一个问题,就直接把我在那边的回答复制过来了。

看到这个问题我也是懵逼了,于是去看了String.intern()的源码:返回在字符串池中存在与之Unicode相同编码的字符串,==判断的结果与两个比较的equals的结果相同。

1、首先明白String s = "abc'' 和 String s = new String("abc")的区别:
String s = "abc''表示从字符串池获取对象并赋值,,如果没有则创建并加入到字符串池,这样可以起到复用的作用,这个过程最多只创建一个对象 字符串abc。
String s = new String("abc") 则会先创建abc,并放入到字符串池,然后创建s对象,因为已经有相同的Unicode编码,所以不会放入到字符串池。

现在分析你的代码:

String a = new StringBuilder("aa").append("计算机").toString();
此过程创建了3个字符串对象,分别是aa, 计算机,aa计算机(toString创建的),aa,计算机已加入到字符串池。aa计算机是toString()得到的结果,此时字符串池并没有Unicode相同的字符串对象,所以也放入到字符串池,此时a对象正好就是aa计算机这个字符串。
System.out.println(a.intern()==a);
所以a.intern()得到正好就是a字符串自己,此处是判断 a == a,返回true。

String b = new StringBuilder().append("计算机").toString();
因为在上一次操作中,计算机这个字符串已经加入到字符串池了,这次并不会产生新的字符串放入到字符串池,b对象是通过StringBuilder的toString()方法得到的新对象。
System.out.println(b.intern()==b);
此处b.intern()得到是a创建时放入字符串池的 计算机,两个不同对象的==判断,肯定是false。

String c = new String("dsd");
这个之前已经说过了,创建了两个对象:字符串dsd 和 字符串c,dsd放入到字符串池。
System.out.println(c.intern()==c);
这里同样是两个不同对象的==判断,所以返回false。

qq_22447717
deadoggy 学习了!!!十分感谢!!
2 年多之前 回复

原理:
因为str1指向的是字符串中的常量,str2是在堆中生成的对象,所以str1==str2返回false
str2调用intern方法,会将str2中值(“string”)复制到常量池中,但是常量池中已经存在该字符串(即str1指向的字符串),所以直接返回该字符串的引用,因此str1==str3返回true。
希望对你有用!

        String str1 = "计算机";//str1指向字符串中的常量
        String str2 = new StringBuilder("计算机").toString();//str2指向堆中生成的对象
        String str3 = str2.intern();//str2调用intern方法,会将str2对象的值("计算机")复制到常量池中,因常量池中已有该字符串(即str1指向的字符串),所以直接返回该字符串的引用,所以str3 == str1返回true;
      System.out.println(str1==str2);
      System.out.println(str1==str3);


        String str4 = "计算机aaa";
        String str5 = new StringBuilder("计算机").append("aaa").toString();//append为添加字符串,以下的道理同上
        String str6 = str5.intern();
        System.out.println(str4==str5);
        System.out.println(str4==str6);

以下是返回结果:

false
true
false
true

qq_22447717
deadoggy 学习了!!谢谢!
2 年多之前 回复
qq_26440221
DaXian-Will 回复Tastill: 如果有就指向,如果没有就生成;前面是指向对象的,后面是是指向字符串的;
2 年多之前 回复
sxlfzhangjie
sxlfzhangjie 楼主说的意思是 a.intern()==a 的比较 ,第一次和第二次是两个程序,第一次没有加append 比较是false, 第二次是单独写的加了append 然后比较久返回true,intern()这个方法是先在常量迟中找 但是找了没有第一次返回false 第二次返回true 为啥加了个append()就发生变化了。
2 年多之前 回复
Tastill
Tastill String str5 = new StringBuilder("计算机").append("aaa").toString();这一步是否已经在常量池中生成了“计算机aaa”这个字符串常量?然后str2.intern()再去引用的时候,其实字符串常量已经存在了。所以直接返回该字符串?
2 年多之前 回复

JDK1.7之前的话,你上面2个输出结果都是true,jdk1.7已经对String的intern方法做了修改。如果你要了解详细,查看下java的源码,这样更加能深入了解。

Csdn user default icon
上传中...
上传图片
插入图片
抄袭、复制答案,以达到刷声望分或其他目的的行为,在CSDN问答是严格禁止的,一经发现立刻封号。是时候展现真正的技术了!
立即提问