对象引用 String引用 基本类型引用 差别
最近遇到一個線上問題,原因是忽略的引用的一些語法,導致出錯,現在記錄一下:
@Testpublic void testList(){List<String> list = new ArrayList<String>();list.add("1");list.add("2");list.add("3");List<String> list2 = new ArrayList<String>();list2.add("1");list2.add("2");//我現在想獲得一個新list,內容是list中有,但list2中沒有的//于是我用了下面的代碼實現List<String> list1 = list;list1.removeAll(list2);System.out.println(list);//[3]---看這個,原始的list也變了System.out.println(list1);//[3]System.out.println(list2);//[1, 2]//很明顯,這里原始list的結果,并不是我想要的,我并不想讓list發生變化//但由于我是直接將list的引用給了list1,其實這個時候list1和list是指向同一個對象的,所以對list1做操作,list也會發生變化,因為這兩個壓根就是一個東西//做個比喻,list就是一個人,他的名字叫list,他有三顆糖,分別1號,2號,3號//那么同時list2就是另外一個人,他的名字叫list2,他有兩顆糖,分別1號,2號//我想要的,是獲得list有,list2卻沒有的糖,我采用了排除法//但我做的操作卻是:List<String> list1 = list;//這個相當于給list另外起了個外號,叫list1,但還是同一個人//所以我拿走list1的糖,其實也就是拿走了list的糖,因為就是一個人}這個問題,對于自定義對象一樣,只給引用,再操作那個新的引用,其實操作的都是一個對象,但是,對于String字符串,卻不是這樣的
@Testpublic void testString(){String a = "1";String b = a;b += "2";System.out.println(a);//1System.out.println(b);//12//看這個操作,照樣是引用,但a卻沒有因為b的改變}擴展總結:
基本類型對象存放在棧中,
引用類型對象數據存放在堆中,
String類型數據存在堆中,但可能存兩份,看下面代碼
第一行:首先在String常量池中查找有沒有字符常量“good”.因為沒有所以創建“good”對象,當執行new String(“good”)時,則在Java的堆中創建一個”good”對象,而a是該對象的引用,因此共計創建2個對象,都在堆中,一個在常量池內,一個在常量池外,兩個good雖然是兩個對象,但卻占一塊內存,都在堆中,只不過是常量池外的good指向常量池中的goog,a引用指的是常量池外的good
第二行:首先在String常量池中查找有沒有字符串常量“good”,有則直接將b作為String常量池中“good”的一個引用,當你重新聲明一個String型變量為“good”時,將使用串池里原來的那個“good”,而不用重新分配內存,也就是說,str與str1將會指向同一塊內存,因此,此時沒有創建任何對象。
第三行:依次在String常量池中查找有沒有字符串常量“good”,有則不進行再次創建,由于這里用了new關鍵字(有new就有對象),所以在Java堆中又創建了一個“good”對象(地址與第一句在堆中創建的地址不同),而c則是這個對象的引用,因此執行此句時,創建了1個對象。
下圖是a,b,c,good,list,list1,list2的存儲位置:
總結
以上是生活随笔為你收集整理的对象引用 String引用 基本类型引用 差别的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: sed 删除某一行_Linux常用命令三
- 下一篇: mapper注入失败,NoSuchBea