問題: 若是Java是用引用來傳遞的話,爲何交換函數(swap)不起做用呢? java
回答: 你的問題引出了Java新手的常犯的錯誤。事實上,一些老手也很難搞清楚這些概念。 函數
Java確實使用對象的引用來作計算的,全部的對象變量都是引用。可是,Java在向方法傳遞參數時傳的不是引用,是值。 spa
以 badSwap() 函數爲例: 對象
1
2
3
4
5
6
|
publicvoidbadSwap(intvar1,intvar2)
{
inttemp = var1;
var1 = var2;
var2 = temp;
}
|
當badSwap方法返回時,被看成參數傳入的變量仍然保持了原來的值不變。若是咱們把傳入的int型變量改成Object型也是同樣的,由於Java經過傳值來傳遞引用的。如今,咱們來看下是哪一個地方搞的鬼: ci
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
publicvoidtricky(Point arg1, Point arg2)
{
arg1.x =100;
arg1.y =100;
Point temp = arg1;
arg1 = arg2;
arg2 = temp;
}
publicstaticvoidmain(String [] args)
{
Point pnt1 =newPoint(0,0);
Point pnt2 =newPoint(0,0);
System.out.println("X: "+ pnt1.x +" Y: "+pnt1.y);
System.out.println("X: "+ pnt2.x +" Y: "+pnt2.y);
System.out.println(" ");
tricky(pnt1,pnt2);
System.out.println("X: "+ pnt1.x +" Y:"+ pnt1.y);
System.out.println("X: "+ pnt2.x +" Y: "+pnt2.y);
}
|
執行這個函數,將獲得如下輸出:
———————————————————-
X: 0 Y: 0
X: 0 Y: 0 get
X: 100 Y: 100
X: 0 Y: 0
———————————————————-
即便是經過值傳遞,tricky函數依然成功地改變了pnt1的值。可是pnt1和pnt2的置換失敗了。這正是最使人困惑的地方。在main()函數當中,pnt1和pnt2僅僅是對象的引用。當你向tricky()函數傳遞pnt1和pnt2參數時,Java僅僅向傳遞任何其餘參數同樣,經過傳值來傳遞引用。這就意味着:傳向函數的引用其實是原始引用的副本。下面的圖一展示了當Java傳遞對象給函數以後,兩個引用指向了同一對象 it
圖一: 當被傳遞給函數以後,一個對象至少存在兩個引用 table
Java複製並傳遞了「引用」的值,而不是對象。所以,方法中對對象的計算是會起做用的,由於引用指向了原來的對象。可是由於方法中對象的引用是「副本」,因此對象交換就沒起做用。如圖2所示,交換動做只對方法中的引用副本起做用了,不影響方法外的引用。因此很差意思,方法被調用後,改變不了方法外的對象的引用。若是要對方法外的對象引用作交換,咱們應該交換原始的引用,而不是它的副本。 import
圖二: 只有傳入函數的引用交換了,原始引用則沒有 stream
我的理解:java參數的傳遞傳的是引用,可是是經過傳值的方式來傳遞引用,也就是說傳過來的參數不是原始對象,而是建立了一個新的對象,只不過這個新的對象與原始對象都是同一個引用。因此副本對象屬性的改變會反應到原始對象,可是副本對象經過從新賦值改變的只是自己的引用。因此對象的交換不起做用,由於改變的只是副本的引用,沒有改變原始對象的引用。