問:當一個對象被看成參數傳遞到一個方法後,此方法可改變這個對象的屬性,並可返回變化後的結果,那麼這裏究竟是值傳遞仍是引用傳遞?
答:是值傳遞。Java 編程語言只有值傳遞參數。當一個對象實例做爲一個參數被傳遞到方法中時,參數的值就是該對象的引用一個副本。指向同一個對象,對象的內容能夠在被調用的方法中改變,但對象的引用(不是引用的副本)是永遠不會改變的。html
Java參數,無論是原始類型仍是引用類型,傳遞的都是副本(有另一種說法是傳值,可是說傳副本更好理解吧,傳值一般是相對傳址而言)。java
不管是什麼語言,要討論參數傳遞方式,就得從內存模型提及,主要是我我的以爲從內存模型來講參數傳遞更爲直觀一些。閒言少敘,下面咱們就經過內存模型的方式來討論一下Java中的參數傳遞。編程
這裏的內存模型涉及到兩種類型的內存:棧內存(stack)和堆內存(heap)。基本類型做爲參數傳遞時,傳遞的是這個值的拷貝。不管你怎麼改變這個拷貝,原值是不會改變的。看下邊的一段代碼,而後結合內存模型來講明問題:編程語言
public class ParameterTransfer { public static void main(String[] args) { int num = 30; System.out.println("調用add方法前num=" + num); add(num); System.out.println("調用add方法後num=" + num); } public static void add(int param) { param = 100; } }
這段代碼運行的結果以下:函數
調用add方法前num=30 調用add方法後num=30
程序運行的結果也說明這一點,不管你在add()方法中怎麼改變參數param的值,原值num都不會改變。ui
下邊經過內存模型來分析一下。url
當執行了int num = 30;這句代碼後,程序在棧內存中開闢了一塊地址爲AD8500的內存,裏邊放的值是30,內存模型以下圖:spa
執行到add()方法時,程序在棧內存中又開闢了一塊地址爲AD8600的內存,將num的值30傳遞進來,此時這塊內存裏邊放的值是30,執行param = 100;後,AD8600中的值變成了100。內存模型以下圖:
3d
地址AD8600中用於存放param的值,和存放num的內存沒有任何關係,不管你怎麼改變param的值,實際改變的是地址爲AD8600的內存中的值,而AD8500中的值並未改變,因此num的值也就沒有改變。
以上是基本類型參數的傳遞方式,下來咱們討論一下對象做爲參數傳遞的方式。指針
先看下邊的示例代碼:
public class ParameterTransfer { public static void main(String[] args) { String[] array = new String[] {"huixin"}; System.out.println("調用reset方法前array中的第0個元素的值是:" + array[0]); reset(array); System.out.println("調用reset方法後array中的第0個元素的值是:" + array[0]); } public static void reset(String[] param) { param[0] = "hello, world!"; } }
運行的結果以下:
調用reset方法前array中的第0個元素的值是:huixin 調用reset方法後array中的第0個元素的值是:hello, world!
當對象做爲參數傳遞時,傳遞的是對象的引用,也就是對象的地址。下邊用內存模型圖來講明。
當程序執行了String[] array = new String[] {"huixin"}後,程序在棧內存中開闢了一塊地址編號爲AD9500內存空間,用於存放array[0]的引用地址,裏邊放的值是堆內存中的一個地址,示例中的值爲BE2500,能夠理解爲有一個指針指向了堆內存中的編號爲BE2500的地址。堆內存中編號爲BE2500的這個地址中存放的纔是array[0]的值:huixin。
當程序進入reset方法後,將array的值,也就是對象的引用BE2500傳了進來。這時,程序在棧內存中又開闢了一塊編號爲AD9600的內存空間,裏邊放的值是傳遞過來的值,即AD9600。能夠理解爲棧內存中的編號爲AD9600的內存中有一個指針,也指向了堆內存中編號爲BE2500的內存地址,如圖所示:
這樣一來,棧內存AD9500和AD9600(即array[0]和param的值)都指向了編號爲BE2500的堆內存。
在reset方法中將param的值修改成hello, world!後,內存模型以下圖所示:
改變對象param的值其實是改變param這個棧內存所指向的堆內存中的值。param這個對象在棧內存中的地址是AD9600,裏邊存放的值是BE2500,因此堆內存BE2500中的值就變成了hello,world!。程序放回main方法以後,堆內存BE2500中的值仍然爲hello,world!,main方法中array[0]的值時,從棧內存中找到array[0]的值是BE2500,而後去堆內存中找編號爲BE2500的內存,裏邊的值是hello,world!。因此main方法中打印出來的值就變成了hello,world!
小結:
不管是基本類型做爲參數傳遞,仍是對象做爲參數傳遞,實際上傳遞的都是值,只是值的的形式不用而已。第一個示例中用基本類型做爲參數傳遞時,將棧內存中的值30傳遞到了add方法中。第二個示例中用對象做爲參數傳遞時,將棧內存中的值BE2500傳遞到了reset方法中。當用對象做爲參數傳遞時,真正的值是放在堆內存中的,傳遞的是棧內存中的值,而棧內存中存放的是堆內存的地址,因此傳遞的就是堆內存的地址。這就是它們的區別。
補充一下,在Java中,String是一個引用類型,可是在做爲參數傳遞的時候表現出來的倒是基本類型的特性,即在方法中改變了String類型的變量的值後,不會影響方法外的String變量的值。關於這個問題,能夠參考以下兩個地址:
http://freej.blog.51cto.com/235241/168676
http://dryr.blog.163.com/blog/static/58211013200802393317600/
我以爲是這兩篇文章中提到的兩個緣由致使的,一個是String實際上操做的是char[],能夠理解爲String是char[]的包裝類。二是給String變量從新賦值後,實際上沒有改變這個變量的值,而是從新new了一個String對象,改變了新對象的值,因此原來的String變量的值並無改變。