- String是内容不可变的,而StringBuffer、StringBuilder都是内容可变的。
- StringBuffer 很多方法带有synchronized关键字,是同步的,数据安全,但效率低;StringBuilder是不同步的,数据不安全,效率高。
- StringBuffer、StringBuilder均继承与AbstractStringBuilder。
- equals()方法
public class StringBufferDemo {
public static void main(String[] args) {
String s = "hello world";
StringBuffer sb1 = new StringBuffer(s);
StringBuffer sb2 = new StringBuffer(s);
System.out.println("s=" + s);
System.out.println("sb1=" + sb1);
System.out.println("sb2=" + sb2);
System.out.println(s.equals(sb1));
System.out.println(s.equals(sb1.toString()));
System.out.println(sb1.equals(sb2));
System.out.println(sb1.toString().equals(sb2.toString()));
}
}
输出:
s=hello world
sb1=hello world
sb2=hello world
false
true
false
true
String的equals
方法因为重写了Object的equals方法,所以可以比较字符串的内容;而StringBuffer因为没重写equals方法,直接继承了Object的equals方法,比较的是引用值是否相等。
StringBuffer类型比较大小,需要先转换成String类型,再比较大小。
sb1.toString()
- replace方法
public class StringBufferDemo {
public static void main(String[] args) {
String s = "java";
StringBuffer sb1 = new StringBuffer("java");
stReplace(s);
sbReplace(sb1);
System.out.println("s=" + s);
System.out.println("sb1=" + sb1);
}
public static void stReplace(String st) {
st.replace("j", "L");
}
public static void sbReplace(StringBuffer sb) {
sb.replace(0,1,"L");
}
}
输出:
s=java
sb1=Lava
因为String类的replace方法返回新的String对象,所以不管stReplace方法内怎么修改对原对象 s 没有任何影响,所以String作为参数传递,效果和基本类型作为参数传递是一样的,形式参数的改变不影响实际参数。而StringBuffer类的replace方法对原对象内容进行修改,所以打印Lava。
那怎么才能修改呢?修改stReplace方法:
public static String stReplace(String st) {
String ret = st.replace("j", "L");
return ret;
}
调用的地方也要修改,s = stReplace(s);
。这次输出就是 s=Lava 。
String、StringBuilder、StringBuffer 使用场景
String
:
适用于少量的字符串操作的情况。如常量的声明,少量的字符串操作。
StringBuilder
:
在单线程情况下,如有大量的字符串操作情况,应该使用StringBuilder。不能使用String"+"来拼接而是使用,避免产生大量无用的中间对象,耗费空间且执行效率低下。如JSON的封装等。
StringBuffer
:
在多线程情况下,如有大量的字符串操作情况,应该使用StringBuffer。如HTTP参数解析和封装等。