深入理解java中的String

今天咱们一起来说说java家族中的使用频率最广泛的String

  • String的特性
    在Sring源码的类注释当中有如下一段话:


    这说明了String的一个重要特性,String 是value不可改变的
    然而String类还使用了final来修饰,表明String的第二个重要特性,** String是不可被继承的**

  • String的定义方法
    一般我们在代码中可以使用如下三种方法定义String对象

String str1 = new String("cat");
String str2 = "cat";
String str3 = "c"+"at";
String str4 = str1+"cat";

第一种方式直接通过关键字new创建对象,毫无疑问str1指向堆内存
第二种直接赋值,Str2指向常量池
第三种通过+号连接赋,最终str3也是指向常量池
第四种是带三种的延伸,这个最终str4是指向堆内存
对于前三种方式大家应该都没什么疑问,那么第四种方法为什么也会指向堆内存呢?欲知真相还需从字节码入手啊!

先来看看几个经典的面试题
  • First
String s1 = "abc";
String s2 = "abc";
System.out.println("s1 == s2 : " + (s1 == s2)); //true

So esay,对吧!

  • Second
String s1 = "abc";
String s2 = new String("abc");
String s3 = new String("abc");
System.out.println("s2 == s3 : " + (s2 == s3)); //false
System.out.println("s1 == s3 : " + (s1 == s3));//false

还是很简单,对么,只要有点基础的人都知道答案呢!

  • Third
 public class Test{
    public static void main(String[] args){
        
        String str1 = "ab" + "cd"; // 
        String str11 = "abcd";
        //System.out.println("str1 == str11 : " + (str1 == str11)); //true
        
        String str2 = "ab"+2;
        String str22 = "ab2";
        //System.out.println("str2 == str22 : " + (str2 == str22)); //true

        final String str3 = "ab"; 
        String str31 = str3+"cd"; 
        String str32 = "abcd";
        //System.out.println("str2 = str3 : " + (str2 == str3)); // false
    }
}

    

这个对新手来说可能就有点不知所措了,来来来,上我们的终极利器,请看下面编译之后的字节码


因为常量的值在编译期间就已经确定了,这里"ab","cd","2"都是常量,所以编译器会自动对String str1 = "ab" + "cd";String str2 = "ab"+2;进行优化,效果相当于String str1 = "abcd";String str2 = "ab2"; 这下你明白为什么结果会是true了吧

  • Fourth
public class Test{
    public static void main(String[] args){     
        String str2 = "ab"; 
        String str3 = "cd"; 
        String str41 = str2 + str3;
        String str42 = "ab"+str3;
        String str43 = str2+"cd";
        String str5 = "abcd";
        //System.out.println("str41 = str5 : " + (str41 == str5)); // false
        //System.out.println("str42 = str5 : " + (str42 == str5)); // false
        //System.out.println("str43 = str5 : " + (str43 == str5)); // false
    }
}

这个跟例三差不多,只不过例三是常量,这里变成了变量,结果就迥然不同,下面我们来看看这段代码的字节码长啥样子



发现虚拟机在处理变量字符串用+号连接的时候是生成了一个StringBuilder对象,然后调用StringBuilder的append方法,最后在调用StringBuilder的toString方法返回。StringBuilder的toString方法源码如下:



这下就彻底明白了吧。它是从新new一个String对象,结果当然为false了!
针对==的总结如下

众所周知,==是判断内存地址的,那么我们判断的话看引用最终指向哪里不就可以了么

  1. new出来的对象是存放在堆内存中的,引用指向堆内存,那么地址肯定是唯一的了,所以结果肯定为false
  2. 直接声明的字符串引用是指向常量池的,而常量池中的相同内容的字符串只有一份,所以结果为true
  3. 通过+号连接的字符串分如下两种情况
    3.1 如果+号两边连接的是常量,那么会在编译期间进行优化,结果同2
    3.2 如果+号两边连接的有变量,不管是new出来的也好,直接声明的也好,java虚拟机执行的时候都会生成一个StringBuilder对象sb,然后调用sb.apend()方法,最后通过sb.toString()返回一个新的字符串。那么此时引用就指向堆内存,结果同1

String.intern()

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

public class Test{
    public static void main(String[] args){
        String s00 = "kvill"; 
        String s11 = new String("kvill"); 
        String s22 = new String("kvill"); 
        System.out.println( s00 == s11 ); //false
      
        s11.intern(); //虽然执行了s1.intern(),但它的返回值没有赋给s1
        s22 = s22.intern(); //把常量池中"kvill"的引用赋给s2 
        System.out.println( s00 == s11); //flase
        System.out.println( s00 == s11.intern() ); //true 说明s11.intern()返回的是常量池中"kvill"的引用
        System.out.println( s00 == s22 ); //true
    }
}

关于String,StringBuffer,StringBuilder的区别
  1. String是值不可变的,每次进行连接操作是都是返回一个新的String对象,StringBuffer,StringBuilder是值可变的,操作是返回的是this
    这也就是为什么在进行大量字符串连接运算时,不推荐使用String,而推荐StringBuffer和StringBuilder。
  2. StringBuffer是线程同步的,安全性高,但执行效率低
    StringBuilder是非线程同步的,安全性低,但执行效率高

这是StringBuffer,StringBuilder append方法的源码




THE END

如有错误之处还请大家不吝赐教,多多指正,共同进步!

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

推荐阅读更多精彩内容

  • java笔记第一天 == 和 equals ==比较的比较的是两个变量的值是否相等,对于引用型变量表示的是两个变量...
    jmychou阅读 1,477评论 0 3
  • 一.你了解String类吗? 想要了解一个类,最好的办法就是看这个类的实现源代码,String类的实现在 \jdk...
    Viking_Den阅读 664评论 0 3
  • 『叮! 深夜我突然想起一年多前的一件事来,但是我不知道该怎么去描述。 就拟人吧。 从前,我很喜欢很喜欢一个男生,但...
    绵花不白阅读 258评论 2 0
  • 拍摄设备 尼康700D 拍摄地点时间 信阳师范学院 晚8:30
    陈凯字东洲阅读 183评论 0 3
  • 曾经是个努力,纯真的小女孩 努力做功课,喜欢看书,参加许多活动,获得许多奖,成绩优秀,只为成为自己想要成为的人 现...
    薄荷味彩铅阅读 181评论 0 0