String类详解
源码:
public final class String
implements java.io.Serializable, Comparable<String>, CharSequence {
/** The value is used for character storage. */
private final char value[];
/** Cache the hash code for the string */
private int hash; // Default to 0
//...
}
String底层是被final关键字修饰的char数组,String类不能被继承,它的成员方法都默认为final方法,字符串一旦创建就不能再修改。
内存分配:
在Java的内存分配中,总共3种常量池,分别是Class常量池、运行时常量池、字符串常量池。
JVM为了提高性能和减少内存的开销,在实例化字符串的时候进行了一些优化:使用字符串常量池。每当创建字符串常量时,JVM会首先检查字符串常量池,如果该字符串已经存在常量池中,那么就直接返回常量池中的实例引用。如果字符串不存在常量池中,就会实例化该字符串并且将其放到常量池中。由于String字符串的不可变性,常量池中一定不存在两个相同的字符串。
对象生成方案:
-
通过字面量赋值创建字符串
如 String s = “test”, 可能创建一个或者不创建对象。先在常量池中查找是否存在相同的字符串(如 String s = “test”),若存在,则将栈中的引用直接指向该字符串;若不存在,则在常量池中生成一个字符串,再将栈中的引用指向该字符串。
-
通过new关键字创建字符串
如 String s = new String(“test”), 至少会创建一个对象,也可能会创建两个对象。因为用到new关键字,肯定会在堆中创建一个String对象,如果字符池中已经存在"test",则不会在字符串池中创建一个String对象,如果不存在,则会在字符串常量池中也创建一个对象。
intern方法:
返回字符串对象的规范化表示形式
-
1.7之前
查询常量池中是否有字符串存在,如果存在,则返回常量池中的引用。如果在常量池找不到对应的字符串,则将字符串拷贝到常量池中。
-
1.7之后
查询常量池中是否有字符串存在,如果存在,则返回常量池中的引用。如果在常量池找不到对应的字符串,则不会再将字符串拷贝到常量池,而只是在常量池中生成一个对原字符串的引用。
常量字符串的拼接操作:
- 常量字符串的“+”操作,编译阶段直接会合成为一个字符串。如string str=”JA”+”VA”,在编译阶段会直接合并成语句String str=”JAVA”,于是会去常量池中查找是否存在”JAVA”,从而进行创建或引用。
- 对于final字段,编译期直接进行了常量替换(而对于非final字段则是在运行期进行赋值处理的)。
final String str1=”ja”;
final String str2=”va”;
String str3=str1+str2;
在编译时,直接替换成了String str3=”ja”+”va”,根据第三条规则,再次替换成String str3=”JAVA” - 常量字符串和变量拼接时(如:String str3=baseStr + “01”;)会调用stringBuilder.append()在堆上创建新的对象。
示例![在这里插入图片描述](https://i-blog.csdnimg.cn/blog_migrate/ee7185257d1061b932f1133f5ce816ac.png)
String s = new String("test")
System.out.println(s == s.intern()) //false
String s9 = new String("1") + new String("1");
s9.intern();
String s10 = "11";
System.out.println(s9 == s9.intern()); //true
System.out.println(s9 == s10); //true
String ss1 = new String("nam");
String ss2 = new String("e");
String ss3 = ss1 + ss2;
String ss4 = ss3.intern();
String ss5 = "name";
System.out.println(ss3 == ss4); //false
System.out.println(ss4 == ss5); //true
其他相关:
- jdk1.7之前,常量池放在方法区,jdk1.7之后,常量池放在堆内存。
- StringBuilder的toString()方法会创建一个新对象,底层为new String()
参考博文: