1、String不可被继承
String 被声明为 final,因此它不可被继承。(Integer 等包装类也不能被继承)
数组被声明为 final,这意味着数组初始化之后就不能再引用其它数组。并且 String 内部没有改变数组的方法,因此可以保证 String 不可变。
2、关于String Pool
字符串常量池(String Pool)保存着所有字符串字面量(literal strings),这些字面量在编译时期就确定。不仅如此,还可以使用 String 的 intern() 方法在运行过程中将字符串添加到 String Pool 中。
当一个字符串调用 intern() 方法时,如果 String Pool 中已经存在一个字符串和该字符串值相等(使用 equals() 方法进行确定),那么就会返回 String Pool 中字符串的引用;否则,就会在 String Pool 中添加一个新的字符串,并返回这个新字符串的引用。
下面为两种创建字符串方式:
String s1 = new String("abcd");
String s2 = new String("abcd");
System.out.println(s1 == s2); // false
String s3 = "bbb";
String s4 = "bbb";
System.out.println(s3 == s4); // true
(1)用new String()创建的字符串,一共会创建两个字符串对象(前提是 String Pool 中还没有 "abcd" 字符串对象)。
- "abcd" 属于字符串字面量,因此编译时期会在 String Pool (串池)中创建一个字符串对象,指向这个 "abcd" 字符串字面量;
- 而使用 new 的方式会在堆中创建一个字符串对象。
(2)采用 "bbb" 这种字面量的形式创建字符串,会自动地将字符串放入 String Pool 中。
3、重要例子
String str1="abcd";
String str2="ab"+"cd";
System.out.println(str1 == str2); // true
String str3="ab";
String str4=str3+"cd";
System.out.println(str1 == str4); // false
首先,第一个str1==str2,由于编译时常量的值就被确定了,这里"ab" "cd"都是常量,因而str2的值在编译时就已经确定,不会进行下面的操作,所以两者相等;
而第二个例子为字符串拼接,首先会创建StringBuilder类,之后用append方法来拼接,再调用StringBuilder的toString()方法在堆中创建对象,而str1在串池中,显然两者不等。