创建字符串对象的方法和内存使用

以下分析基于HotSpot虚拟机
注意jdk1.7以上的两个变化:

  1. 虚拟机内存中,常量池由方法区转移到堆。
  2. 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”

posted @ 2020-03-03 15:47  chzhyang  阅读(516)  评论(0编辑  收藏  举报