conact()
方法只接受字符串类型的参数,参数不能为空;conact()
底层是依靠Arrays.copy()方法实现的先看个例子,代码如下:
public class StringContactExample2 {
public static void main(String[] args) {
System.out.println(new StringContactExample2().testContact("abc", null));
System.out.println(new StringContactExample2().testContact2("abc", null));
}
public String testContact(String a, String b) {
return a += b;
}
public String testContact2(String a, String b) {
return a.concat(b);
}
}
这个例子的运行结果如下:
image.png
可以通过查看字节码和JDK源码来比较二者的不同,将上面的代码使用javac StringContactExample2.java
编译,然后使用javap -c StringContactExample2
,可以看到对应的字节码内容。
testContact()方法字节码如下所示,从第0行可以看出,编译器做了优化,运算符重载“+”在字节码层面生成了一个StringBuilder对象,然后依靠append()方法进行连接。
public java.lang.String testContact(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
testContact2的字节码如下所示:
public java.lang.String testContact2(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
contact()方法的源码实现如下所示,可以看出是依赖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);
}