String、StringBuffer、StringBuilder 的区别
Java 平台提供了两种类型的字符串:String 和 StringBuffffer/StringBuilder,它们可以储存和操作字符串。
String ,是只读字符串,也就意味着 String 引用的字符串内容是不能被改变的。
每次对 String 类型进行改变的时候,都会生成一个新的 String 对象,然后将指针指向新的 String 对
象。
StringBuffffer/StringBuilder 类,表示的字符串对象可以直接进行修改。StringBuilder 是 Java 5 中引入的,它
和 StringBuffffer 的方法完全相同,区别在于它是在单线程环境下使用的,因为它的所有方面都没有被
synchronized 修饰,因此它的效率也比 StringBuffffer 要高。
StringBuffffer 每次都会对 StringBuffffer 对象本身进行操作,而不是生成新的对象并改变对象引用。相同情况下使用 StirngBuilder 相比使用 StringBuffffer 仅能获得 10%~15% 左右的性能提升,但却要冒
多线程不安全的风险。
对于三者使用的总结?
操作少量的数据 = String 。
这个也是实际编码较为经常使用的方式。
单线程操作字符串缓冲区下操作大量数据 = StringBuilder 。
甚至有时,我们为了避免每个线程重复创建 StringBuilder 对象,会通过 ThreadLocal + StringBuilder
的方式,进行对 StringBuilder 的重用。
多线程操作字符串缓冲区下操作大量数据 = StringBuffffer
实际场景下,我们基本不太会出现,多线程操作同一个 StringBuffffer 对象。
String s = new String("xyz") 会创建几个对象?
首先,在 String 池内找,找到 "xyz" 字符串,不创建 "xyz" 对应的 String 对象,否则创建一个对象。
然后,遇到 new 关键字,在内存上创建 String 对象,并将其返回给 s ,又一个对象。
所以,总共是 1 个或者 2 个对象。
String 为什么是不可变的?
简单的来说,String 类中使用 final 关键字字符数组保存字符串。代码如下:
// String.java
private final char[] value;
所以 String 对象是不可变的。
而 StringBuilder 与 StringBuffffer 都继承自 AbstractStringBuilder 类,在 AbstractStringBuilder 中也是使用字符
数组保存字符串 char[] value ,但是没有用 final 关键字修饰。代码如下:
// AbstractStringBuilder.java
char[] value;
所以这两种对象都是可变的。
StringTokenizer 是什么?
StringTokenizer ,是一个用来分割字符串的工具类。
示例代码如下:StringTokenizer st = new StringTokenizer(”Hello World”);
while (st.hasMoreTokens()) {
System.out.println(st.nextToken());
}
输出如下:
Hello
World