Java的String理解

String对象方式的创建方式

   public static void main(String[] args) {
        //1.new的方式创建
        String s = new String("abc");
        //2. 字面量字符串赋值
        String s1 = "bcd";
        //3. 多个字面量字符串拼接
        String s2 = "abc" + "de";
        //4. 含有变量的拼接
        String s3 = s1 + "de";
    }

查看main的字节码

  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=3, locals=5, args_size=1
         0: new           #2                  // class java/lang/String
         3: dup
         4: ldc           #3                  // String abc
         6: invokespecial #4                  // Method java/lang/String."<init>":(Ljava/lang/String;)V
         9: astore_1
        10: ldc           #5                  // String bcd
        12: astore_2
        13: ldc           #6                  // String abcde
        15: astore_3
        16: new           #7                  // class java/lang/StringBuilder
        19: dup
        20: invokespecial #8                  // Method java/lang/StringBuilder."<init>":()V
        23: aload_2
        24: invokevirtual #9                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        27: ldc           #10                 // String de
        29: invokevirtual #9                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        32: invokevirtual #11                 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
        35: astore        4
        37: return
      LineNumberTable:
        line 12: 0
        line 14: 10
        line 16: 13
        line 18: 16
        line 19: 37
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      38         0  args         [Ljava/lang/String;
           10      28        1     s           Ljava/lang/String;
           13      25        2    s1          Ljava/lang/String;
           16      22        3    s2          Ljava/lang/String;
           37       1         4    s3          Ljava/lang/String;

//1.new的方式创建
String s = new String("abc");
创建的方式,可以看到
0: new #2 // class java/lang/String
这是在堆上创建了一个对象

//2. 字面量字符串赋值
String s1 = "bcd";
//3. 多个字面量字符串拼接
String s2 = "abc" + "de";
10: ldc #5 // String bcd
12: astore_2
13: ldc #6 // String abcde
15: astore_3
对象s2,s3直接从常量池的取值

ldc 表示从常量池提取数据并压入操作数栈

abcde并没有直接完整的形式出现,但是通过下面的常量池是可以发现abcde是存在常量池的
#6 = String #37 // abcde

所以: 若字符串通过字符串字面量的方式生成,是字符串的对象是从常量池中获取的.

1.必须全部是字符串字面量 2.可以有多个,单个字面量赋值是多个的特例

//4. 含有变量的拼接
String s3 = s1 + "de";

       16: new           #7                  // class java/lang/StringBuilder
        19: dup
        20: invokespecial #8                  // Method java/lang/StringBuilder."<init>":()V
        23: aload_2
        24: invokevirtual #9                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        27: ldc           #10                 // String de
        29: invokevirtual #9                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        32: invokevirtual #11                 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;

通过字节码发现只要有(非fianl的)变量的参与拼接,那么jvm会通过新建StringBuilder对象的方式来创建
通过StringBuilder.append()的方法追踪源码,发现是不会共享String的char数组的.
若在s1前面添加final,
//2. 字面量字符串赋值
final String s1 = "bcd";
//4. 含有final变量的拼接
String s3 = s1 + "de";
那么等价于String s3 = "bcd" + "de";

继续变:
String s = new String("abc");
final String s1 = s;
String s3 = s1 + "de";
此时就算s1是final 的,但是因为s不是final 的很有可能变动,所以s1,s3都是通过在堆上新建StringBuilder对象生成字符串对象

对应的常量池

Constant pool:
   #1 = Methodref          #13.#32        // java/lang/Object."<init>":()V
   #2 = Class              #33            // java/lang/String
   #3 = String             #34            // abc
   #4 = Methodref          #2.#35         // java/lang/String."<init>":(Ljava/lang/String;)V
   #5 = String             #36            // bcd
   #6 = String             #37            // abcde
   #7 = Class              #38            // java/lang/StringBuilder
   #8 = Methodref          #7.#32         // java/lang/StringBuilder."<init>":()V
   #9 = Methodref          #7.#39         // java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
  #10 = String             #40            // de
  #11 = Methodref          #7.#41         // java/lang/StringBuilder.toString:()Ljava/lang/String;
  #12 = Class              #42            // com/cqs/learning/java/StringDemo
  #13 = Class              #43            // java/lang/Object
  #14 = Utf8               <init>
  #15 = Utf8               ()V
  #16 = Utf8               Code
  #17 = Utf8               LineNumberTable
  #18 = Utf8               LocalVariableTable
  #19 = Utf8               this
  #20 = Utf8               Lcom/cqs/learning/java/StringDemo;
  #21 = Utf8               main
  #22 = Utf8               ([Ljava/lang/String;)V
  #23 = Utf8               args
  #24 = Utf8               [Ljava/lang/String;
  #25 = Utf8               s
  #26 = Utf8               Ljava/lang/String;
  #27 = Utf8               s1
  #28 = Utf8               s2
  #29 = Utf8               s3
  #30 = Utf8               SourceFile
  #31 = Utf8               StringDemo.java
  #32 = NameAndType        #14:#15        // "<init>":()V
  #33 = Utf8               java/lang/String
  #34 = Utf8               abc
  #35 = NameAndType        #14:#44        // "<init>":(Ljava/lang/String;)V
  #36 = Utf8               bcd
  #37 = Utf8               abcde
  #38 = Utf8               java/lang/StringBuilder
  #39 = NameAndType        #45:#46        // append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
  #40 = Utf8               de
  #41 = NameAndType        #47:#48        // toString:()Ljava/lang/String;
  #42 = Utf8               com/cqs/learning/java/StringDemo
  #43 = Utf8               java/lang/Object
  #44 = Utf8               (Ljava/lang/String;)V
  #45 = Utf8               append
  #46 = Utf8               (Ljava/lang/String;)Ljava/lang/StringBuilder;
  #47 = Utf8               toString
  #48 = Utf8               ()Ljava/lang/String;
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,589评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,615评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,933评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,976评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,999评论 6 393
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,775评论 1 307
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,474评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,359评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,854评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,007评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,146评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,826评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,484评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,029评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,153评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,420评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,107评论 2 356