我们从下面三个方面进行阐述:
-
一、可变性
-
String类长度是不可变的,它使用final修饰的字符数组保存字符串(private final charvalue[])
-
StringBuffer和StringBuilder长度是可变的,它们都继承自AbstractStringBuilder类,在AbstractStringBuilder类中使用未被final修饰的字符数组保存字符串(char[] value;)并且StringBuffer和StringBuilder的构造方法都是基于父类AbstractStringBuilder实现的
-
二、线程安全性
-
String对象是
不可变
的,也就是说String对象一旦被创建后就无法改变了,直到对象被销毁,String也可以理解为常量,线程安全
。 -
StringBuffer对方法加了同步锁或者调用方法的时候加了
同步
修饰,所以是线程安全
的,但是性能会非常受影响。 -
StringBuilder
没有对方法加同步锁
,所以是非线程安全
的,但是性能比较好。 -
String 是不可变的对象, 因此在每次对 String 类型进行改变的时候,都会生成一个新的 String 对象,然后将指针指向新的 String 对象,所以经常改变内容的字符串如果使用 String ,会在内存中生成很多无用的对象, JVM 的 GC 会频繁工作,性能就会降低。
-
三、性能
-
使用 StringBuffer 类和StringBuilder类时,每次都会对 StringBuffer/StringBuilder 对象本身进行操作,而不是生成新的对象并改变对象引用。所以多数情况下推荐使用 StringBuffer /StringBuilder,性能比较好,特别是字符串对象经常改变的情况下。但是根据统计,相同使用情况下StringBuilder仅比StringBuffer多10%~15%左右的性能提升。
- 如果在频繁进行字符串运算(如拼接、替换、删除等),并且运行在
多线程
环境中,则可以考虑使用StringBuffer,例如XML解析、HTTP参数解析和封装。 - 如果在频繁进行字符串运算(如拼接、替换、和删除等),并且运行在
单线程
环境中,则可以考虑使用StringBuilder,如SQL语句的拼装、JSON封装等。 -
使用总结
- 如果要操作少量的数据用
String
单线程
操作字符串缓冲区下操作大量数据用StringBuilder
多线程
操作字符串缓冲区下操作大量数据用StringBuffer
String 类的常用方法都有那些?
- indexOf():返回指定字符的索引。
- charAt():返回指定索引处的字符。
- replace():字符串替换。
- trim():去除字符串两端空白。
- split():分割字符串,返回一个分割后的字符串数组。
- getBytes():返回字符串的 byte 类型数组。
- length():返回字符串长度。
- toLowerCase():将字符串转成小写字母。
- toUpperCase():将字符串转成大写字符。
- substring():截取字符串。
- equals():字符串比较。