创建字符串对象的方法和内存使用
以下分析基于HotSpot虚拟机
注意jdk1.7以上的两个变化:
- 虚拟机内存中,常量池由方法区转移到堆。
- String.intern(),如果常量池存在等于此String对象的字符串,则返回该常量池中对象的引用;否则,在常量池中生成一个对原字符串的引用,并返回该引用,而不会再将字符串拷贝到常量池(jdk1.6)。
new String
在堆上创建字符串对象,调用String类的有参构造函数,内部可以简化为如下代码:
String temp = "hello"; // 在常量池中
String str = new String(temp); // 在堆上
如果常量池中不存在"hello",这段代码创建了2个String对象,temp指向在常量池中的,str指向堆上的,而str内部的char value[]则指向常量池中的char value[]。
否则,只会创建str。
通过字面量赋值创建字符串
String s3 = "a"
先在常量池中查找是否存在相同的字符串,若存在,则将栈中的引用直接指向该字符串;
若不存在,则在常量池中生成一个字符串,再将栈中的引用指向该字符串。
字符串常量拼接
编译阶段直接会合成为一个字符串。
string str=”JA”+”VA”
在编译阶段会直接合并成语句String str=”JAVA”,于是会去常量池中查找是否存在”JAVA”,从而进行创建或引用。
字符串变量拼接
String s1=baseStr + “01”
调用stringBuilder.append()在堆上创建新的对象。
例如下面的代码:
String str = "pingtouge";
for(int i=0; i<1000; i++) { str = str + i;}
编译器会优化为:
String str = "pingtouge";
for(int i=0; i<1000; i++)
{
str = (new StringBuilder(String.valueOf(str))).append(i).toString();
}
String s2 = new String("a")+new String("a");
先执行两个new string,在执行两个string变量的拼接
对于final字段,编译期直接进行了常量替换(而对于非final字段则是在运行期进行赋值处理的)。
final String str1=”a”;
final String str2=”va”;
String str3=str1+str2;
在编译时,直接替换成了String str3=”a”+”va”,再次替换成String str3=”ava”