1. String str=new String("abc")和String str="abc"的字符串“abc”都是存放在堆中,而不是存在
棧中。
2. 其實在在java中有一個“字符數據池”的內存管理機制。
3. String str="abc",執行這句話時,會先去“字符數據池”搜索時候有“abc”這個字符串,如果有
,則將字符串的首地址賦值給str,如果沒有,生成一個新的字符串“abc”並且將首地址賦值給str;
4. String str=new String("abc"),執行這句話時,不會考慮時候已經存在了“abc”這個字符串,而
是直接生成一個新的字符串“abc”並將首地址賦值給str,注意“abc”並不放在“字符數據池”中;
5. 由以上分析可知,String str="abc"和效率要高於String str=new String("abc"),因為如果有重復
的字符串時,第一種方式可以節省空間。
6. 下面舉例說明一下,好好看看結果,仔細分析原因,上面已經說明的很清楚了:
public class Test{
public static void main(String args[]){
String s1=new String("abc");//直接在堆中生成新的“abc”
String s2=new String("abc");//直接在堆中生成新的“abc”
String s3="abc";//先去“字符數據池”搜索時候有“abc”這個字符串,如果有,則
將字符串的首地址賦值給s3,如果沒有,則在“字符數據池”中生成一個新的字符串“abc”並且將首地
址賦值給s3;
String s4="abc";//去“字符數據池”搜索時發現了上一步生成的“abc”這個字符串
,把該字符串首地址賦值給s4,這時其實s3和s4指向同一個字符數據池中的“abc”
System.out.println(s1==s2);
System.out.println(s1==s3);
System.out.println(s2==s3);
System.out.println(s3==s4);
}
}
結果:
false
fasle
false
true
另外:例如:
String str1=”java”; //指向字符串池
String str2=”blog”; //指向字符串池
String s=str1+str2; //s是指向堆中值為"javablog"的對象,+運算符會在堆中建立來兩個String對象,這兩個對象的值分別是"java" "blog". 也就是說從字符串池中復制這兩個值,然后在堆中創建兩個對象,然后再建立對象s,然后將"javablog"的堆地址賦給s. 這句共創建了?個String 對象!
System.out.println(s==”javablog”); //結果是false。
Jvm確實對型如String str1=”java”;的String對象放在常量池里,但是它是在編譯時那么做的,而String s=str1+str2;是在運行時刻才能知道,也就是說str1+str2是在堆里創建的,所以結果為false了。
如果改成一下兩種方式:
String s="java" + "blog"; //直接將"javablog"放入字符串池中,System.out.println(s==”javablog”); 的結果為true, 這個句子創建了?個String對象
String s=str1+ "blog"; //不放入字符串池,而是在堆中分配,System.out.println(s==”javablog”); 的結果為False, 這個句子創建了?個String對象
總結
綜上所述,創建字符串有兩種方式:兩種內存區域(pool,heap)
1," " 引號創建的字符串在字符串池中
2,new,new創建字符串時首先查看池中是否有相同值的字符串,如果有,則拷貝一份到堆中,然后返回堆中的地址;如果池中沒有,則在堆中創建一份,然后返回堆中的地址(注意,此時不需要從堆中復制到池中,否則,將使得堆中的字符串永遠是池中的子集,導致浪費池的空間)!
另外,對字符串進行賦值時,如果右操作數含有一個或一個以上的字符串引用時,則在堆中再建立一個字符串對象,返回引用;如String s=str1+ "blog";