Integer和int及String的總結
秉承著總結髮表是最好的記憶,我把之前遇到的問題在這裡總結和大家分享一下,希望大家共同進步:
一.Integer和int
首先說下自動拆裝箱,基本資料型別轉換為包裝型別的過程叫裝箱,反之則是拆箱,其中最特殊也是面試經常遇到的是Integer,下面總結說明,話不多說,show the code,先猜一下下面這6個題的答案:
1 Integer a1=new Integer(10); 2 Integer b1=new Integer(10); 3 System.out.println("1:"+(a1==b1));//1 4 5Integer a2=128; 6 Integer b2=128; 7 System.out.println("2:"+(a2==b2));//2 8 9 Integer a3=5; 10 Integer b3=5; 11 System.out.println("3:"+(a3==b3));//3 12 13 int a4=3; 14 Integer b4=new Integer(3); 15 System.out.println("4:"+(a4==b4));//4 16 17 Integer a5=new Integer(40); 18 Integer b5=new Integer(40); 19 Integer c5=new Integer(0); 20 System.out.println("5.1:"+(a5==b5));//5.1 21 System.out.println("5.2:"+(a5==b5+c5));//5.2 22 23 Integer a6=30; 24 Integer b6=new Integer(30);25 System.out.println("6:"+(a6==b6));//6
讓我們看下對應的答案,要養成良好的習慣,先猜完在看答案 >O<
---------------我------是-----答-----案-----分----割-----線------------------
1:false 2:false 3:true 4:true 5.1:false 5.2:true 6:false
看了答案之後是不是有點意外,如果你答錯了,請繼續往下看,下面的具體分析可能會對你有所幫助,但是如果你都答對了並且已經很清楚原因了,那麼請你也繼續往下看,因為有可能我的分析有不對的地方,還要請你指正出來,不廢話了,開始分析:
1.
1 Integer a1=new Integer(10); 2 Integer b1=new Integer(10); 3 System.out.println("1:"+(a1==b1));//1 false
這個答案為false可能你猜到了,因為包裝類屬於引用型別,使用new是在堆中建立了一個物件,而且此處用的比較方法時“==”,比較的是記憶體地址,所以兩個物件的記憶體不一樣很正常,如果改為equels比較,結果就為true;
2.
1 Integer a2=128; 2 Integer b2=128; 3 System.out.println("2:"+(a2==b2));//2 false
這個是自動轉換,jvm會自動轉換,實際就等同與Integer a=new Integer(128),然後就和上道題一樣的解釋了,明白了吧。
3.
1 Integer a3=5; 2 Integer b3=5; 3 System.out.println("3:"+(a3==b3));//3 true
看到答案是不是感覺懵X了,上面不是說的好好的,按照上面的解釋,答案就是false啊,但實際卻為true,彆著急,我來解釋,這個就比較特殊了,涉及到jvm的包裝型別的快取機制了,如果Integer要裝箱,會先判斷值,如果是在-128到127之間,則不會new Integer(X),而是直接從快取中(常量池)獲取的,所以上述的兩個值都是取至快取中的同一個值,所以引用地址是一樣的,答案為true,網上關於這一塊將的比較詳細,如果有想更細緻的研究,可以搜尋網上的文件,還有原始碼參考,我就不在Ctrl C V了。
4.
1 int a4=3; 2 Integer b4=new Integer(3); 3 System.out.println("4:"+(a4==b4));//4 true
說到這個,還有點故事,有一次電話面試,面試官問了我一個這樣的問題,當時的我真的是靠懵,但是不知道是我沒聽清,還是面試官沒說明白,說把上述的3換成128,答案就是false,為此,我自己親自驗證,然後又網上搜索,最後確定這個說法是錯誤的,不管把值換成多少,答案都是true;當Integer和int做“==”比較時,會自動的拆箱,把Integer轉換為int型別,然後在值比較。
5.
1 Integer a5=new Integer(40); 2 Integer b5=new Integer(40); 3 Integer c5=new Integer(0); 4 System.out.println("5.1:"+(a5==b5));//5.1 false 5 System.out.println("5.2:"+(a5==b5+c5));//5.2 true
如果前面的都看了,這個5.1的答案你可能猜對了,但是這個5.2的答案看著是不是又懵X了,彆著急,我再來解釋下5.2,當Integer型別做運算時,會自動拆箱,把Integer型別轉換為int型別,如果是用“==”做比較,則比較的是值,所以答案是true。
6.
1 Integer a6=30; 2 Integer b6=new Integer(30); 3 System.out.println("6:"+(a6==b6));//6 false
看完前面的種種情況之後,相信你會猜對這個題的答案了,不僅是猜,而且也能解釋個一二了,我就再解釋下吧(解釋的有點多........),Integer b6=30;會直接從常量池中取,而不會在堆中new Integer(30),所以兩個引用的地址不一樣,用“==”比較時答案就為false。
看完這些相信大家已經很明白了,然後最後在給大家留個小作業,考驗下大家,請看題:
1 int a7=30; 2 Integer b7=30; 3 System.out.println("7:"+(a7==b7));//7
----------答--------案--------在--------此---------
不用懷疑自己,這個相信你已經會很自信的用自己的知識解釋答案了,是的,答案為true,我沒有給各位看官下套套。。。。。。只是讓大家更堅信自己了~
二.String的常量池
剛才上面說到Integer的常量池了,大家肯定不禁想到String的常量池了,是不是正好可以建立關聯記憶了,是的,我也忍不住想總結一下了,那就囉唆幾句吧。
先拋個磚......
問:這個會建立幾個“abc”?
答:不知道。
這個答案實在是高啊,我也不知道會建立幾個,扯淡吧;其實正確的答案是一個或者是兩個,因為這個得看常量池,如果常量池中已經存在“abc”這個字串了,則會在堆中建立一次,
如果常量池中沒有“abc”,則會在堆中new一個“abc”,然後在常量池中再建立一個“abc”,
這個就是標準的解釋。
以後如果碰到這種類似的題都可以按照這種解釋,不管是 還是上面new的方式,jvm都會先檢查常量池中是否有該字串,如果有,則 這種方式就不會再建立新的字串了,如果是 這種方式,則也會先判斷常量池,如果常量池中已經存在,則會只在堆中建立一次;所有的這類題目都可以按照這種思路解釋,如果出題的話,可以挖出很多大坑,希望大家以後要小心,下面出個題趁熱打鐵給大家練練手吧,請看題:
1 String m="abc"; 2 String n=new String("abc"); 3 System.out.println("1:"+(m==n));//1 false 4 String s="abc"; 5 System.out.println("2:"+(m==s));//2 true
答案在圖中的註釋,如果有跳不出坑的童鞋,可以給我留言,我會一一的給你們囉唆的明明白白的。
然後由於聯想記憶法,我又想到了String、StringBuffer和StringBuilder的知識了,啊......好煩躁,真想忍不住的在給大家囉唆幾句,想想還是算了吧,真的是沒完沒了T.T
上述的總結中,如果有不一樣的觀點,或錯誤,歡迎大神給予指正哦~