用 Google photo 搜索关键词「String」,有兴趣的朋友可以试试,感觉发现了新大陆…
Java 中有 3 种字符串的拼接方式,了解这三种拼接方式的实现,将有益于提高自己的代码质量。
本文主要讲解 String 对象的三种拼接方式,以及它们之间的效率对比。
三种方式
1 2 3 4 5 6 7 8 9 10 11
| // 方式 1 String s = "Hello"; s += "Hello";
// 方式 2 StringBuilder stringBuilder = new StringBuilder("Hello"); stringBuilder.append("Hello");
// 方式 3 StringBuffer stringBuffer = new StringBuffer("Hello"); stringBuffer.append("Hello");
|
自己写个主函数,将上面 3 个方法循环执行 10000 次;
执行时间 t1 > t3 > t2,也就是说 StringBuilder.append() 拼接最快,String += 最慢。
源码解读
- StringBuilder.append() 拼接
StringBuilder.java
1 2 3 4 5 6 7 8 9
| public final class StringBuilder extends AbstractStringBuilder implements java.io.Serializable, CharSequence
@Override public StringBuilder append(String str) { super.append(str); // 调用父类 append 方法 return this; }
|
AbstractStringBuilder.java
1 2 3 4 5 6 7 8 9
| public AbstractStringBuilder append(String str) { if (str == null) return appendNull(); int len = str.length(); ensureCapacityInternal(count + len); str.getChars(0, len, value, count); // 调用 String 的 getChars 方法 count += len; return this; }
|
String.java
1 2 3 4 5 6 7 8 9 10 11 12
| public void getChars(int srcBegin, int srcEnd, char dst[], int dstBegin) { if (srcBegin < 0) { throw new StringIndexOutOfBoundsException(srcBegin); } if (srcEnd > value.length) { throw new StringIndexOutOfBoundsException(srcEnd); } if (srcBegin > srcEnd) { throw new StringIndexOutOfBoundsException(srcEnd - srcBegin); } System.arraycopy(value, srcBegin, dst, dstBegin, srcEnd - srcBegin); // 本地方法 }
|
从上面 3 段代码可知,StringBuilder.append 方法调用的是父类 AbstractStringBuilder.append,父类 append 方法调用了 String 的本地方法 System.arraycopy 实现
StringBuilder.java
1 2 3 4 5 6 7 8 9 10
| public final class StringBuffer extends AbstractStringBuilder implements java.io.Serializable, CharSequence
@Override public synchronized StringBuffer append(String str) { toStringCache = null; super.append(str); return this; }
|
StringBuffer.append 方法与 StringBuilder.append 方法类似,不同的是,append 方法加上了 synchronized 锁,说明该方法适用于多线程环境,但是加锁的过程需要耗时,所以执行时间比 StringBuilder.append 慢。
对下面代码进行编译,生成的 class 文件使用命令 javap -c Main.class,可以反编译得到图片中的字节码。
可以看到 += 实际调用的是 StringBuilder.append() 方法,所以速度会比 StringBuilder 慢。
1 2 3 4 5
| public static void main(String[] args) { String s = "Hello"; s = s + " World"; System.out.println(s); }
|
总结
所以在需要拼接字符串的场合,尽量适用 StringBuilder.append() 方法,多线程环境下则推荐 StringBuffer.append() 方法,而 String += 拼接方式任何场合都不建议使用。