JAVA String 相加编译器发生了什么?

501 查看

javapackage com.spring;
public class Day01 {




    public static void main(String[] args) {

         String  b="aa"+"bb";

    }

}

String b="aa"+"bb"编译器发生了什么?
通过查看class文件可以知道:

// Compiled from Day01.java (version 1.5 : 49.0, super bit)
public class com.spring.Day01 {

  // Method descriptor #6 ()V
  // Stack: 1, Locals: 1
  public Day01();
    0  aload_0 [this]
    1  invokespecial java.lang.Object() [8]
    4  return
      Line numbers:
        [pc: 0, line: 10]
      Local variable table:
        [pc: 0, pc: 5] local: this index: 0 type: com.spring.Day01

  // Method descriptor #15 ([Ljava/lang/String;)V
  // Stack: 1, Locals: 2
  public static void main(java.lang.String[] args);
    0  ldc <String "aabb"> [16]
    2  astore_1 [b]
    3  return
      Line numbers:
        [pc: 0, line: 17]
        [pc: 3, line: 19]
      Local variable table:
        [pc: 0, pc: 4] local: args index: 0 type: java.lang.String[]
        [pc: 3, pc: 4] local: b index: 1 type: java.lang.String
}

在编译的过程中间就将加直接处理成了"aabb";

但是如果是这样:

javapackage com.spring;


public class Day01 {




    public static void main(String[] args) {
       String b="bb";
         String c ="aa"+b;

    }

}

查看class文件:

// Compiled from Day01.java (version 1.5 : 49.0, super bit)
public class com.spring.Day01 {

  // Method descriptor #6 ()V
  // Stack: 1, Locals: 1
  public Day01();
    0  aload_0 [this]
    1  invokespecial java.lang.Object() [8]
    4  return
      Line numbers:
        [pc: 0, line: 10]
      Local variable table:
        [pc: 0, pc: 5] local: this index: 0 type: com.spring.Day01

  // Method descriptor #15 ([Ljava/lang/String;)V
  // Stack: 3, Locals: 3
  public static void main(java.lang.String[] args);
     0  ldc <String "bb"> [16]
     2  astore_1 [b]
     3  new java.lang.StringBuilder [18]
     6  dup
     7  ldc <String "aa"> [20]
     9  invokespecial java.lang.StringBuilder(java.lang.String) [22]
    12  aload_1 [b]
    13  invokevirtual java.lang.StringBuilder.append(java.lang.String) : java.lang.StringBuilder [25]
    16  invokevirtual java.lang.StringBuilder.toString() : java.lang.String [29]
    19  astore_2 [c]
    20  return
      Line numbers:
        [pc: 0, line: 16]
        [pc: 3, line: 17]
        [pc: 20, line: 19]
      Local variable table:
        [pc: 0, pc: 21] local: args index: 0 type: java.lang.String[]
        [pc: 3, pc: 21] local: b index: 1 type: java.lang.String
        [pc: 20, pc: 21] local: c index: 2 type: java.lang.String
}

可以看出是new StringBuffer("aa").append("bb").toString();
而且可以看到astore_1 [b]表示在栈内存生成了一个引用变量指向堆内存的"bb"。而"aa"则没有引用变量指向。(可以查看JAVA指令集了解这些内容)
非常量字会串相加时,由于相加的变量中存放的是字符串的地址引用,
因为在编译时无法确切地知道其他具体的值,也就没有办法对其进行优化处理,这时为了
达到连接的效果,其内部采用了 StringBuffer 的机制进行处理。

堆内存java虚拟机可以直接使用。