参考答案
concat()
方法只接受字符串类型的参数,参数不能为空;concat()
底层是依靠Arrays.copy()方法实现的- 如果有必要的话,”+”会对参与连接的变量进行toString();
- ”+“底层是依靠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);
}
}
这个例子的运行结果如下:
可以通过查看字节码和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);
}
参考资料
本号专注于后端技术、JVM问题排查和优化、Java面试题、个人成长和自我管理等主题,为读者提供一线开发者的工作和成长经验,期待你能在这里有所收获。
019:使用String的conact()方法和“+”做字符串拼接有什么不同