可能很多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);
}
}
这个程序很好理解,因为在解析的时候,"abc"被存储在常量池中,str1和str2的引用都是指向常量池中的"abc"。所以str1和str2引用是相同的。
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); } }
这里由于编译器做了优化,编译器会先把字符串拼接,再在常量池中查找这个字符串是否存在,如果存在,则让变量直接引用该字符串。所以str1和str3引用也是相同的。
str4的对象不是显式赋值的,编译器会在堆中重新分配一个区域来存储它的对象数据。所以str1和str4的引用是不一样的。
3.知道字符串怎么存储和赋值的,就不难理解在下面这些例子到底创建了几个String对象了
例1:
String str = "abc";
这里毫无疑问是一个对象
例2:
String a = "abc"; String b = "abc";
这里呢?有的人可能会说有两个,但是这里其实只有一个对象.这里a和b都应用对象池里的同一个对象.
例3:
String str = "ab" + "cd";
这里也只有一个,因为这里是字面量的连接,它被优化了,在Java编译的时候就可以确定,然后直接在对象池取,它等同于
String str = "abcd";
例4:
String str1 = "ab"; String str2 = str1 + "cd";
这里总共有3个对象
例5:
String str = new String("abc");
这里是两个对象,一个是String类型的引用对象,一个是引用对象所指向的实例对象