概览
String被声明为final,因此它不可被继承。
public final class String
implements java.io.Serializable, Comparable<String>, CharSequence {
/** The value is used for character storage. */
private final char value[];
内部使用char数组存储数据,该数组被声明为final,这意味着value数组初始化之后就不能再引用其他数组。并且String内部没有改变value数组的方法,因此可以保证String不可变。
不可变的好处
- 可以缓存hash值
- String Pool的需要
如果一个String对象已经被创建过了,那么就会从String Pool中取得引用。只有String是不可变的,才可能使用String Pool。 - 安全性
String经常作为参数,String不可变性可以保证参数不可变。 - 线程安全
String不可变性天生具备线程安全,可以在多个线程中安全地使用。
String,StringBuffer and StringBuilder
- 可变性
- String不可变
- StringBuffer和StringBuilder可变
- 线程安全
- String不可变,因此是线程安全的
- StringBuilder不是线程安全的
- StringBuffer是线程安全的,内部使用synchronized进行同步
String.intern()
使用String.intern()可以保证相同用内容的字符串变量引用同一的内存对象。
下面示例中,s1和s2采用new String()的方式新建了两个不同对象,而s3是通过s1.intern()方法取得一个对象引用。intern()首先把s1引用的对象放到String Pool(字符串常量池)中,然后返回这个对象引用。因此s3和s1引用的是同一个字符串常量池的对象。
String s1 = new String("aaa");
String s2 = new String("aaa");
System.out.println(s1 == s2); // false
String s3 = s1.intern();
System.out.println(s1.intern() == s3); // true
如果是采用"bbb"这种使用双引号的形式创建字符串实例,会自动地将新建的对象放入String Pool中。
String s4 = "bbb";
String s5 = "bbb";
System.out.println(s4 == s5); // true
在Java 7之前,字符串常量池被放在运行时常量池中,它属于永久代。而在Java 7,字符串常量池被移到Native Method中。这是因为永久代的空间有限,在大量使用字符串的场景下会导致OutOfMemoryError错误。