深入理解String和StringBuilder

前言#

我偶尔会在问答里面看一看谁有问题,如果我知道就回答一下。今天偶然看到一个问题,去掉提问的代码部分,问题就是:

String对象的intern()方法得到的对象,为什么与String对象有时候不相等?

我看到这个问题,我也是懵逼了,今天就写个笔记复习一下。

正文#

首先,这个String对象的intern()方法是干什么用的呢?看一下源码的注释:

/**
* 英文注释好长,这里就简单翻译一下
* 返回一个这个String对象的权威代表(请注意,这里返回的是代表,没说返回是自己)
* 有一个字符串池,专门用来维持String对象,当intern方法被调用的时候,返回和他equals方法相同的String对象,如果没有,就把这个String添加到池中,再把这个String对象返回(也就是说这个情况,返回了自己)
* 
* s.intern() == t.intern(),只有在s.equals(t)等于true
*/     
public native String intern();

看来这个方法和equals关系密切,所以再看一下equals方法:

public boolean equals(Object anObject) {
    // 先判断是否是同一个对象
        if (this == anObject) {
            return true;
        }
        // 判断是否是String类型
        if (anObject instanceof String) {
            String anotherString = (String) anObject;
            int n = count;
            // 判断字符串的长度是否相等
            if (n == anotherString.count) {
                int i = 0;
                // 判断每一个位置的字符是否相等
                while (n-- != 0) {
                    if (charAt(i) != anotherString.charAt(i))
                            return false;
                    i++;
                }
                return true;
            }
        }
        return false;
    }

equals判断的仅仅的是字符串的内容,所以只要内容相同,在字符串池中都不会重复添加。

结合我们已经对字符串的了解,我们可以总结出一下几点:

1、字符串池中,只包含唯一内容的字符串。
2、字符串池,提供了相同字符串之间的复用机制,防止不同字符串创建多个对象。

这个时候突然想起来刚接触Java时的一个面试题:

String s = "abc" 和 String s = new String("abc") 的区别

这个问题大对数都能答对:

String s = "abc" 是先使用字符串池中的abc对象,如果没有创建abc并添加到字符串池中,这个逻辑和intern()方法是完全一样的, 所以这种使用方法也是推荐的使用方法。

String s = new String("abc") ,一开始的过程和第一种是一样的,同样是先从字符串池中获取,然后根据情况添加或者返回。但是new操作符,会返回一个新的String对象,也就是说,返回的String对象并不是abc。但是这种方法会出现内存的浪费,所以并不推荐使用。

为了验证我们的想法,我们运行一个小demo:

public class Main {
    
    public static void main(String[] args){
        // 注意这里通过创建StringBuilder,已经创建了111,并加入到字符串池
        String s1 = new StringBuilder("111").toString();
    // 这里还是通过相同的方式,看看是否返回了跟s1相同的对象
        String s2 = new StringBuilder("111").toString();
        // 直接从字符串池中得到对象
        String s3 = "111";
        String s4 = "111";
       
        System.out.println(s1.equals(s2));
        System.out.println(s1 == s2);

        System.out.println(s3.equals(s4));
        System.out.println(s3 == s4);
    }

}

首先我们使用了两个StringBuilder来拼接字符串,看看得到的结果,然后直接从字符串池中去取,看看得到是不是同一个对象。

true
false
true
true

第一个结果是s1.equals(s2) =true,这个没有疑问,对比内容必然是相同的。
第二个结果是s1 == s2 得到false,说明是s1和s2是相同内容的不同对象。

为什么不是相同对象呢?看一下StringBuilder的toString()方法:

@Override
public String toString() {
        // Create a copy, don't share the array
   return new String(value, 0, count);
}

竟然是一个new String,怪不得对象是不相同的。

第三个结果s3.equals(s4) = true, 这个没有疑问,对比内容必然是相同的。
第四个结果s3 == s4,一样是true,说明得到确实是相同的对象。

总结#

有了刚才的验证,我们基本上可以这么理解:

"abc" , 我们可以看做是单例模式,这个abc只创建一次,可以复用。
例如 String s = "abc", StringBuilder.append("abc"),实际上使用的都是同一个字符串对象。

并且我们知道了平时使用字符串的几个小细节:

1、String的equals()方法判断的内容相同,不是判断是否是相同对象。
2、StringBuilder的toString()方法,会创建新的字符串对象并返回,这个还是有优化空间的。

我们把之前学到的内容又重新复习了一遍,还找到了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
?

这个问题你能够帮他解答吗?

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,530评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,403评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,120评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,770评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,758评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,649评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,021评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,675评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,931评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,751评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,410评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,004评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,969评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,042评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,493评论 2 343

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,567评论 18 399
  • 一、String 类 1、定义: 1、从概念上讲,java字符串就是Unicode字符序列。每个用双引号括起来的字...
    玉圣阅读 1,557评论 0 1
  • Tip:笔者马上毕业了,准备开始 Java 的进阶学习计划。于是打算先从 String 类的源码分析入手,作为后面...
    石先阅读 11,990评论 16 58
  • 糖尿病分为一型和二型,临床上绝大多数糖尿病患者都属于二型糖尿病,到今天为止,几乎所有的医学论著,医生和营养师都都在...
    花语88阅读 278评论 0 0
  • 笑与泪最后都会连带着冰冷的身躯卧进沃土之中。在那个万念俱灰的封闭逼仄的空间里,臭虫不会钻进我们的大脑,窥视我...
    取个帅气的昵称拔阅读 185评论 0 0