Java到底是傳引用還是傳值?

問題: 如果Java是用引用來傳遞的話,為什麼交換函數(swap)不起作用呢?

回答: 你的問題引出了Java新手的常犯的錯誤。事實上,一些老手也很難搞清楚這些概念。

Java確實使用對象的引用來做計算的,所有的對象變量都是引用。但是,Java在向方法傳遞參數時傳的不是引用,是值。

以 badSwap() 函數為例:

Java到底是傳引用還是傳值?


當badSwap方法返回時,被當作參數傳入的變量仍然保持了原來的值不變。如果我們把傳入的int型變量改為Object型也是一樣的,因為Java通過傳值來傳遞引用的。現在,我們來看下是哪個地方搞的鬼:

Java到底是傳引用還是傳值?


執行這個函數,將得到以下輸出:

Java到底是傳引用還是傳值?

即使是通過值傳遞,tricky函數依然成功地改變了pnt1的值。但是pnt1和pnt2的置換失敗了。這正是最令人困惑的地方。在main()函數當中,pnt1和pnt2僅僅是對象的引用。當你向tricky()函數傳遞pnt1和pnt2參數時,Java僅僅向傳遞任何其他參數一樣,通過傳值來傳遞引用。這就意味著:傳向函數的引用實際上是原始引用的副本。下面的圖一展現了當Java傳遞對象給函數之後,兩個引用指向了同一對象

Java到底是傳引用還是傳值?

圖一: 當被傳遞給函數之後,一個對象至少存在兩個引用

Java複製並傳遞了“引用”的值,而不是對象。因此,方法中對對象的計算是會起作用的,因為引用指向了原來的對象。但是因為方法中對象的引用是“副本”,所以對象交換就沒起作用。如圖2所示,交換動作只對方法中的引用副本起作用了,不影響方法外的引用。所以不好意思,方法被調用後,改變不了方法外的對象的引用。如果要對方法外的對象引用做交換,我們應該交換原始的引用,而不是它的副本。

Java到底是傳引用還是傳值?

圖二: 只有傳入函數的引用交換了,原始引用則沒有


分享到:


相關文章: