Java究竟是傳引用仍是傳值?

問題: 若是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參數的傳遞傳的是引用,可是是經過傳值的方式來傳遞引用,也就是說傳過來的參數不是原始對象,而是建立了一個新的對象,只不過這個新的對象與原始對象都是同一個引用。因此副本對象屬性的改變會反應到原始對象,可是副本對象經過從新賦值改變的只是自己的引用。因此對象的交換不起做用,由於改變的只是副本的引用,沒有改變原始對象的引用。

相關文章
相關標籤/搜索