一、分析
对于一个字符串进行拼接有三种方法:加号、concat方法、及StringBuiler或StringBuffer。
1."+"方法拼接字符串
str += "c";等效于:str = new StringBuffer(str).append("c").toString();
虽然编译器对字符串加号做了优化,它会用StringBuffer的append方法进行追加。再是通过toString方法转换成String字符串的。
它与纯粹的append方法是不同的:
一是每次都要创建一个StringBuilder对象;
二是每次执行完毕都要调用toString方法将其转换为字符串。
2.concat方法拼接
concat方法的源码:
public string concat(String str){ int otherLen = str.length(); //如果追加的字符串长度为0,则返回字符串本身 if(otherLen == 0){ return this; } //字符串数组,容纳的是新字符串的字符 char buf[] = new char[count + otherLen]; //取出原字符串放到buf数组中 getChars(0, count, buf, 0); //追加的字符串转化成字符串数组,添加到buf中 str.getChars(0, otherLen, buf, count); //赋值字符数组,产生一个新的字符串 return new String(0, count + otherLen, buf); }
从整体上看就是一个数组的拷贝,虽然在内存中的处理都是原子性操作,速度非常快,注意看最后的return语句,每次的concat操作都会创建一个新的String对象,这就是concat速度慢下来的原因。
3.appned方法拼接
StringBuilder的appned方法字节由父类的AbstractStringBuilder实现,代码如下:
public AbstractStringBuilder append(String str){ //如果是null值,则把null作为字符串处理 if(str == null)str = "null"; int len = str.length(); //字符串的长度为0,则返回自身 if(len == 0)return this; int newCount = count + len; //追加后的字符串组长度是否超过当前值 if(newCount > value.length) expandCapacity(newCount);//加长,并作数组拷贝 //字符串复制到目标数组 str.getChars(0, len, value, count); count = newCount; return this; }
整个append方法都在做字符数组处理,加长,然后数组拷贝,这些都是基本的数据处理,没有新建任何对象,所以速度也就最快了!
二、场景
看看如下代码:
public static void doWithStringBuffer(){ StringBuilder db = new StringBuilder("a"); for(int I = 0; I < 50000; I ++){ sb.append("c"); //str += "c"; //str = str.concat("c"); } String str = sb.toString(); }
1.StringBuffer的append方法执行时间是0毫秒,时间非常短暂;
2.contact方法次之,由上分析,每次的concat操作都需要创建一个String对象,它创建了5万个String对象;
3.加法拼接,每次创建一个StringBuilder对象,并执行完毕调用toString()方法转换。它创建了5万个StringBuilder对象,toString转换5万次。
三、建议
三者的实现方法不同,性能也就不同,但是并不表示一定要使用StringBuilder,这是因为“+”非常符合我们的编程习惯,适合人类阅读,大多数情况下,我们使用加号操作。
只有在系统系能临界(如在性能“增长一分则太长”的情况下)的时候,才考虑使用concat或append方法。而且很多时候系统80%的性能是消耗在20%的代码上的,我们的精力应该更多的投入到算法和结构上。