019:使用String的conact()方法和“+”做字符串拼接有什么不同

参考答案

  1. concat()方法只接受字符串类型的参数,参数不能为空;
  2. concat()底层是依靠Arrays.copy()方法实现的
  3. 如果有必要的话,”+”会对参与连接的变量进行toString();
  4. ”+“底层是依靠StringBuilder实现的,Java编译器在字节码层面做了优化;

知识点梳理

先看个例子,代码如下:

public class StringConcatExample2 {

    public static void main(String[] args) {
        System.out.println(new StringConcatExample2().testConcat("abc", null));
        System.out.println(new StringConcatExample2().testConcat2("abc", null));

//        System.out.println(new StringContactExample2().testContact("abc", "ddd"));
//        System.out.println(new StringContactExample2().testContact2("abc", "ddd"));
    }

    public String testConcat(String a, String b) {
        return a += b;
    }

    public String testConcat2(String a, String b) {
        return a.concat(b);
    }
}

这个例子的运行结果如下:
屏幕快照 2019-03-07 上午10.28.38.png

可以通过查看字节码和JDK源码来比较二者的不同,将上面的代码使用javac StringConcatExample2.java编译,然后使用javap -c StringConcatExample2,可以看到对应的字节码内容。

testConcat()方法字节码如下所示,从第0行可以看出,编译器做了优化,运算符重载“+”在字节码层面生成了一个StringBuilder对象,然后依靠append()方法进行连接。

 public java.lang.String testConcat(java.lang.String, java.lang.String);
    Code:
       0: new           #10                 // class java/lang/StringBuilder
       3: dup
       4: invokespecial #11                 // Method java/lang/StringBuilder."<init>":()V
       7: aload_1
       8: invokevirtual #12                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      11: aload_2
      12: invokevirtual #12                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      15: invokevirtual #13                 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
      18: dup
      19: astore_1
      20: areturn

testConcat2的字节码如下所示:

  public java.lang.String testConcat2(java.lang.String, java.lang.String);
    Code:
       0: aload_1
       1: aload_2
       2: invokevirtual #14                 // Method java/lang/String.concat:(Ljava/lang/String;)Ljava/lang/String;
       5: areturn

concat()方法的源码实现如下所示,可以看出是依赖Arrays.copy方法来进行数据的移动。

public String concat(String str) {
    int otherLen = str.length(); //这里可以看出,如果str未null,则会直接报NPE
    if (otherLen == 0) {
        return this;
    }
    int len = value.length;
    char buf[] = Arrays.copyOf(value, len + otherLen);
    str.getChars(buf, len);
    return new String(buf, true);
}

参考资料

  1. https://stackoverflow.com/questions/47605/string-concatenation-concat-vs-operator

本号专注于后端技术、JVM问题排查和优化、Java面试题、个人成长和自我管理等主题,为读者提供一线开发者的工作和成长经验,期待你能在这里有所收获。
javaadu

Share

javaadu

IT从业者,终身学习者

You may also like...

发表评论

电子邮件地址不会被公开。 必填项已用*标注