1、各自简介
String:字符串常量,final修饰。其中,String类源码中有 一个char类型数组 private final char value[],可以看到是final类型的,因此,是不可变长度的。
StringBuffer:线程安全的可变字符序列。StringBuffer 上的主要操作是 append 和 insert 方法,可重载这些方法,以接受任意类型的数据。它的append方法调用的是父类AbstractStringBuilder的append方法。
public AbstractStringBuilder append(String str) {
//新增加的数据为null则直接返回字符串"null"
if (str == null) str = "null";
int len = str.length();//新增加数据的长度为len
ensureCapacityInternal(count + len);//扩容:count+len ;The count is the number of characters used.
str.getChars(0, len, value, count);//getChars方法-->将字符从此字符串复制到目标字符数组。将原来的数据复制到扩容后的数组中
count += len;
return this;
}
public void ensureCapacity(int minimumCapacity) {
if (minimumCapacity > 0)
ensureCapacityInternal(minimumCapacity);
}
private void ensureCapacityInternal(int minimumCapacity) {
// overflow-conscious code
if (minimumCapacity - value.length > 0)
expandCapacity(minimumCapacity);
}
void expandCapacity(int minimumCapacity) {
int newCapacity = value.length * 2 + 2;//char[] value;value是AbstractStringBuilder用来保存字符串内容的char类型数组
if (newCapacity - minimumCapacity < 0)
newCapacity = minimumCapacity;
if (newCapacity < 0) {
if (minimumCapacity < 0) // overflow
throw new OutOfMemoryError();
newCapacity = Integer.MAX_VALUE;
}
value = Arrays.copyOf(value, newCapacity);
}
如上,是StringBuffer如何实现扩容的。
StringBuilder:一个可变的字符序列。建议优先采用该类,因为在大多数实现中,它比 StringBuffer 要快。
2.三者在执行速度方面的比较:
StringBuilder > StringBuffer > String
3.String <(StringBuffer,StringBuilder)的原因
String是“字符串常量”,是不可改变的对象。由于这种机制,每当用String操作字符串时,实际上是在不断的创建新的对象,而原来的对象就会变为垃圾被GC回收掉,效率大大降低。
而StringBuffer与StringBuilder就不一样了,他们是字符串变量,是可改变的对象,每当我们用它们对字符串做操作时,实际上是在一个对象上操作的,这样就不会像String一样创建一些而外的对象进行操作了,当然速度就快了。
4.StringBuilder与 StringBuffer
StringBuilder:线程非安全的
StringBuffer:线程安全的
StiringBuffer跟StringBuilder类似,只不过为了实现同步,很多方法使用Synchronized修饰,如下面的方法:
当我们在字符串缓冲区被多个线程使用时,JVM不能保证StringBuilder的操作是安全的,虽然他的速度最快,但是可以保证StringBuffer是可以正确操作的。当然大多数情况下就是我们是在单线程下进行的操作,所以大多数情况下是建议用StringBuilder而不用StringBuffer的,就是速度的原因。