你知道String对"+"做了什么吗

下面有段简单的代码

public class StringPlusTest{
    public static void main(String[] args) {
        String s1 = "aaa";
        String s2 = "bbb";
        String s = "ccc" +s1 + s2 +"ddd";
    }
}

使用 javap -c .\StringPlusTest.class反编译一下,得到

Compiled from "StringPlusTest.java"
public class com.epoint.codetuning.test.StringPlusTest {
  public com.epoint.codetuning.test.StringPlusTest();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public static void main(java.lang.String[]);
    Code:
       0: ldc           #2                  // String aaa
       2: astore_1
       3: ldc           #3                  // String bbb
       5: astore_2
       6: new           #4                  // class java/lang/StringBuilder
       9: dup
      10: invokespecial #5                  // Method java/lang/StringBuilder."<init>":()V
      13: ldc           #6                  // String ccc
      15: invokevirtual #7                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      18: aload_1
      19: invokevirtual #7                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      22: aload_2
      23: invokevirtual #7                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      26: ldc           #8                  // String ddd
      28: invokevirtual #7                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      31: invokevirtual #9                  // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
      34: astore_3
      35: return
}

其中,

ldc 把常量池中的项压入栈
astore_1 将引用类型或returnAddress类型值存入局部变量1
astore_2 将引用类型或returnAddress类型值存入局部变量2
new 创建一个新对象
dup 复制栈顶部一个字长内容
invokespecial 根据编译时类型来调用实例方法
aload_1 从局部变量1中装载引用类型值
aload_2 从局部变量2中装载引用类型值
astore_3 将引用类型或returnAddress类型值存入局部变量3

更多指令见JVM指令手册

对于Java来说,这段代码原理上应该是:

public class StringPlusTest{
    public static void main(String[] args) {
        String s1 = "aaa";
        String s2 = "bbb";
        String s = new StringBuilder().append("ccc").append(s1).append(s2).append("ddd").toString();
    }
}

由此可见,Java中使用“+”拼接字符串的实现原理是通过建立临时StringBuilder对象调用append和toString方法实现。

对上面代码做一些修改

public class StringPlusTest{
    public static void main(String[] args) {
        String s1 = "aaa";
        String s2 = null;
        String s = "ccc" +s1 + s2 +"ddd";
        System.out.println(s);
    }
}

结果是什么?

运行可以得到如下:

cccaaanullddd

查看StringBuilder源码

@Override
public StringBuilder append(String str) {
    super.append(str);
    return this;
}

该方法调用继承父类AbstractStringBuilder的方法,再去父类中查看

 public AbstractStringBuilder append(String str) {
        if (str == null)
            return appendNull();
        int len = str.length();
        ensureCapacityInternal(count + len);
        str.getChars(0, len, value, count);
        count += len;
        return this;
}
private AbstractStringBuilder appendNull() {
    int c = count;
    ensureCapacityInternal(c + 4);
    final char[] value = this.value;
    value[c++] = 'n';
    value[c++] = 'u';
    value[c++] = 'l';
    value[c++] = 'l';
    count = c;
    return this;
}

appendNull方法简单来说就是容量+4,追加null字符串。

所以结果为”cccaaanullddd”。

本作品采用《CC 协议》,转载必须注明作者和本文链接
讨论数量: 0
(= ̄ω ̄=)··· 暂无内容!

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!