經(jīng)常面試會(huì)被問到這兩個(gè)的區(qū)別,比如String s = new String("abc")創(chuàng)建了幾個(gè)對象,String s = "abc"又是創(chuàng)建了幾個(gè)對象
ps: String s = new String("abc")創(chuàng)建了1個(gè)或2個(gè)對象,String s = "abc"創(chuàng)建了一個(gè)或0個(gè)對象
String s = new String("abc")的創(chuàng)建過程
- 系統(tǒng)先在字符串常量池里面尋找是否有一個(gè)"abc"的字符串,
- 如果有的話,則在堆中復(fù)制一個(gè)該字符串,并且將堆中的引用指向s,這個(gè)時(shí)候系統(tǒng)只創(chuàng)建了一個(gè)對象,即堆中的對象;
- 如果沒有的話,則會(huì)先在字符串常量池中先創(chuàng)建一個(gè)字符串為"abc"的常量,然后再復(fù)制到堆里面,最后將堆所在的地址指向s,這個(gè)時(shí)候創(chuàng)建了兩個(gè)對象;
String s = "abc"的創(chuàng)建過程
- 系統(tǒng)先在字符串中尋找是否存在"abc"的常量
- 如果存在,則直接將該"abc"在常量池中的地址指向s,這個(gè)時(shí)候,系統(tǒng)沒有創(chuàng)建新對象。
- 如果不存在,則在常量池中新建一個(gè)"abc"并放入常量池里面,然后再返回該地址,這個(gè)時(shí)候,系統(tǒng)創(chuàng)建了一個(gè)對象。
例子
String s = "abc";
System.out.println(s == "abc"); // true
因?yàn)閟 和 "abc" 都是指常量池里面"abc" 的地址,所有true;
String s = new String("abc");
System.out.println(s == "abc"); // false
因?yàn)閟 指向的是堆里面新建的對象的地址,而"abc"指向的是常量池里面的地址,因?yàn)椴坏取?/p>
String s = new String("abc");
String s1 = new String("abc");
System.out.println(s == s1); // false
s和s1都是在堆中新建了不同的對象,雖然內(nèi)容一樣,但是兩則是位于堆中不同地址的空間,所以是不相等的。
String s = "a" + "b";
System.out.println(s == "ab"); // true
此處虛擬機(jī)會(huì)做優(yōu)化,會(huì)在常量池里面尋找"a" + "b" 結(jié)果后的字符串即"ab",所以兩者都是對映常量池中"ab"的地址
String s1 = "a";
String s2 = "b";
String str = s1 + s2;
System.out.println(str == "ab"); // false
雖然s1和s2各自指的是常量池里面"a","b"的引用,但是string在做加法或者subString、replace等方法的時(shí)候,實(shí)際上返回的是new String()的結(jié)果,因此str指向的是堆中的地址,所以不相等。
String s = "abc";
String ss = new String("abc").intern();
System.out.println(s == ss); // true
雖然此處ss是new出來的新對象,但是由于調(diào)用了intern方法,這個(gè)方法會(huì)返回常量池中相等值的字符串的地址,所以最后ss指向的是常量池中"abc"的地址,所以相等。