java字符串池和字符串堆內存分配

1. String str=new String("abc")和String str="abc"的字符串「abc」都是存放在堆中,而不是存在java

棧中。
 2. 其實在在java中有一個「字符數據池」的內存管理機制。
 3. String str="abc",執行這句話時,會先去「字符數據池」搜索時候有「abc」這個字符串,若是有spa

,則將字符串的首地址賦值給str,若是沒有,生成一個新的字符串「abc」而且將首地址賦值給str;
 4. String str=new String("abc"),執行這句話時,不會考慮時候已經存在了「abc」這個字符串,而對象

是直接生成一個新的字符串「abc」並將首地址賦值給str,注意「abc」並不放在「字符數據池」中;
 5. 由以上分析可知,String str="abc"和效率要高於String str=new String("abc"),由於若是有重複blog

的字符串時,第一種方式能夠節省空間。
 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
trueclass

 

另外:例如:效率

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";

相關文章
相關標籤/搜索