三分鐘理解Java中字符串(String)的存儲和賦值原理
可能很多java的初學者對String的存儲和賦值有迷惑,以下是一個很簡單的測試用例,你只需要花幾分鐘時間便可理解。
1.在看例子之前,確保你理解以下幾個術語:
棧 :由JVM分配區域,用于保存線程執行的動作和數據引用。棧是一個運行的單位,Java中一個線程就會相應有一個線程棧與之對應。
堆 :由JVM分配的,用于存儲對象等數據的區域。
常量池 :在編譯的階段,在堆中分配出來的一塊存儲區域,用于存儲顯式 的String,float或者integer.例如String str="abc"; abc這個字符串是顯式聲明,所以存儲在常量池。
2.看這個例子,用JDK5+junit4.5寫的例子,完全通過測試
import static org.junit.Assert.assertNotSame;
import static org.junit.Assert.assertSame;
import org.junit.Test;
/**
* @author Heis
*
*/
public class StringTest{
@Test
public void testTheSameReference1(){
String str1="abc";
String str2="abc";
String str3="ab"+"c";
String str4=new String(str2);
//str1和str2引用自常量池里的同一個string對象
assertSame(str1,str2);
//str3通過編譯優化,與str1引用自同一個對象
assertSame(str1,str3);
//str4因為是在堆中重新分配的另一個對象,所以它的引用與str1不同
assertNotSame(str1,str4);
}
}
import static org.junit.Assert.assertSame;
import org.junit.Test;
/**
* @author Heis
*
*/
public class StringTest{
@Test
public void testTheSameReference1(){
String str1="abc";
String str2="abc";
String str3="ab"+"c";
String str4=new String(str2);
//str1和str2引用自常量池里的同一個string對象
assertSame(str1,str2);
//str3通過編譯優化,與str1引用自同一個對象
assertSame(str1,str3);
//str4因為是在堆中重新分配的另一個對象,所以它的引用與str1不同
assertNotSame(str1,str4);
}
}
- 第一個斷言很好理解,因為在編譯的時候,"abc"被存儲在常量池中,str1和str2的引用都是指向常量池中的"abc"。所以str1和str2引用是相同的。
- 第二個斷言是由于編譯器做了優化,編譯器會先把字符串拼接,再在常量池中查找這個字符串是否存在,如果存在,則讓變量直接引用該字符串。所以str1和str3引用也是相同的。
- str4的對象不是顯式賦值的,編譯器會在堆中重新分配一個區域來存儲它的對象數據。所以str1和str4的引用是不一樣的。

程序員的一生其實可短暫了,這電腦一開一關,一天過去了,嚎;電腦一開不關,那就成服務器了,嚎……
posted on 2010-01-03 18:59 Heis 閱讀(8295) 評論(0) 編輯 收藏 所屬分類: 雜七雜八