qq_22447717
deadoggy
采纳率80%
2017-08-07 15:30

Java中StringBuilder构造的String.intern问题

5
  • string
  • java
  • stringbuilder
已采纳

下面一段代码中
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条回答

  • u011315960 珠穆朗玛小王子 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。

    点赞 3 评论 复制链接分享
  • chaoyangsun 孙朝阳 4年前

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

    点赞 评论 复制链接分享
  • admin_CMZ admin_CMZ 4年前

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

    点赞 评论 复制链接分享
  • wojiushiwo945you 毕小宝 4年前

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

    点赞 评论 复制链接分享
  • u013254779 猿小牛 4年前

    StringBuilder的构造参数为aa

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

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

    点赞 评论 复制链接分享
  • sinat_38913556 sinat_38913556 4年前

为你推荐