谈谈StringBuffer和StringBuilder

时间:2022-11-10 15:42:38

(1) 速度

在执行速度方面的比较:StringBuilder > StringBuffer > String

①String 是不可变的对象(String类源码中存放字符的数组被声明为final), 因此在每次对 String 类型进行改变的时候其实都等同于生成了一个新的 String 对象,然后将指针指向新的 String 对象,所以经常改变内容的字符串最好不要用 String ,因为每次生成对象都会对系统性能产生影响,特别当内存中无引用对象多了以后, JVM 的 GC 就会开始工作,那速度是一定会相当慢的。

② StringBuffer与StringBuilder,他们是字符串变量,是可改变的对象,每当我们用它们对字符串做操作时,实际上是在一个对象上操作的,不像String一样创建一些对象进行操作,所以速度就快了。

③在某些特别情况下,String 对象的字符串拼接其实是被 JVM 解释成了 StringBuffer 对象的拼接,所以这些时候 String 对象的速度并不会比 StringBuffer 对象慢,而特别是以下的字符串对象生成中, String 效率要比 StringBuffer 快:

String S1 = “This is only a” + “ simple” + “ test”;
StringBuffer Sb = new StringBuilder(“This is only a”).append(“ simple”).append(“ test”);

你会很惊讶的发现,生成 String S1 对象的速度简直太快了,而这个时候 StringBuffer 居然速度上根本一点都不占优势。其实这是 JVM 的一个把戏,在 JVM 眼里,这个

String S1 = “This is only a” + “ simple” + “test”;

其实就是:

String S1 = “This is only a simple test”; 

所以当然不需要太多的时间了。

(2)线程安全性

StringBuilder:线程非安全的
StringBuffer:线程安全的

StringBuilder一个可变的字符序列是5.0新增的。此类提供一个与 StringBuffer 兼容的 API,但不保证同步。
当我们在字符串缓冲区被多个线程使用是,JVM不能保证StringBuilder的操作是安全的,虽然他的速度最快,但是可以保证StringBuffer是可以正确操作的。
当然大多数情况下就是我们是在单线程下进行的操作,所以大多数情况下是建议用StringBuilder而不用StringBuffer的,就是速度的原因。
如果可能,建议优先采用该类,因为在大多数实现中,它比 StringBuffer 要快。两者的方法基本相同。

(3)使用策略

1.如果要操作少量的数据用 = String
单线程操作字符串缓冲区 下操作大量数据 = StringBuilder
多线程操作字符串缓冲区 下操作大量数据 = StringBuffer

2. 不要使用String类的"+"来进行频繁的拼接,因为那样的性能极差的,应该使用StringBuffer或StringBuilder类,这在Java的优化上是一条比较重要的原则。

(4)速度测试源码

我们测试下拼接N条UUID的耗时

①N等于3W
String方法耗时:26765ms
StringBuffer方法耗时:27ms
StringBuilder方法耗时:24ms

②N等于300W
StringBuffer方法耗时:3337ms
StringBuilder方法耗时:2768ms

测试代码

public class TestDemo {
public static void main(String[] args) { Long start = System.currentTimeMillis(); //①String方法拼接字符串
String str1 = UUID.randomUUID().toString();
int n = 30000;
for (int i = 0; i < n; i++) {
str1 = str1 + UUID.randomUUID().toString();
}
System.out.print("String长度:"+str1.length());
System.out.println(" String拼接耗时:"+(System.currentTimeMillis() - start)+"ms");
start = System.currentTimeMillis();
//②StringBuffer方法拼接字符串
StringBuffer sBuffer = new StringBuffer();
for (int i = 0; i <= n; i++) {
sBuffer.append(UUID.randomUUID().toString());
}
System.out.print("String长度:"+sBuffer.toString().length());
System.out.println(" StringBuffer拼接耗时:"+(System.currentTimeMillis() - start)+"ms");
start = System.currentTimeMillis();
//②StringBuilder方法拼接字符串
StringBuilder sBuilder = new StringBuilder();
for (int i = 0; i <= n; i++) {
sBuilder.append(UUID.randomUUID().toString());
}
System.out.print("String长度:"+sBuilder.toString().length());
System.out.println(" StringBuilder拼接耗时:"+(System.currentTimeMillis() - start)+"ms");
}
}