Java中StringBuilder构造的String.intern问题

下面一段代码中
public static void main(String[] argv){
String a = new StringBuilder("aa").append("计算机").toString();
System.out.println(a.intern()==a);
String b = new StringBuilder().append("计算机").toString();
System.out.println(b.intern()==b);
String c = new String("dsd");
System.out.println(c.intern()==c);

}

为何结果是
    true
    false
    false 

而不是
    false
    false
    fasle
?

8个回答

看到这个问题我也是懵逼了,于是去看了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。

String.intern():
存在于.class文件中的常量池,在运行期被JVM装载,并且可以扩充。String的intern()方法就是扩充常量池的一个方法;当一个String实例str调用intern()方法时,Java查找常量池中是否有相同Unicode的字符串常量,如果有,则返回其的引用,如果没有,则在常量池中增加一个Unicode等于str的字符串并返回它的引用
String s = "a";和String s1 = new String("a");的差别

首先,我认为根源是StringBuilder的构造参数为常量串"aa",这点是关键。这个常量串是存储在常量池中的。而StringBuilder的append操作是将字符串存储在它的成员变量中的。
其次,使用intern方法获取的必定都是常量池中的字符串,只有跟它比较的对象也是常量池中的字符串的情况下==操作才会为true.
最后,c.intern()==c为false,关键是c是一个普通的字符串对象,但是c.intern()=="dsd"就为真了。二者都是真正的常量池中的字符串。

StringBuilder的构造参数为aa

刚刚又看来一会儿,也没明白怎么回事儿,看字节码两段代码没什么不同。toString方法创建的字符串,为什么二者完全不同呢?
才疏学浅,也没想明白。

首先你得明白String和StringBuilder的区别 这个问题就不难了

a.intern()、a都是实例引用,所以为true;
c.intern()是实例引用,c是实例,所以为false.

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