參考文章:http://blog.sina.com.cn/s/blog_59ca2c2a0100qhjx.html java參數有值類型和引用類型兩種。所以java參數的傳值也就從這兩個方面分析。 從記憶體模型來說參數傳遞更為直觀一些,這裡涉及到兩種類型的記憶體:棧記憶體(stack)和堆記憶體(heap) ...
參考文章:http://blog.sina.com.cn/s/blog_59ca2c2a0100qhjx.html java參數有值類型和引用類型兩種。所以java參數的傳值也就從這兩個方面分析。 從記憶體模型來說參數傳遞更為直觀一些,這裡涉及到兩種類型的記憶體:棧記憶體(stack)和堆記憶體(heap)。 基本類型作為參數傳遞時,傳遞的是這個值的拷貝。無論你怎麼改變這個拷貝,原值是不會改變的。
引用類型作為參數傳遞時,傳遞的是引用地址。所以在改變引用類型的值時,原值會改變。(String類型特殊,單獨說)
基本類型作為參數傳遞
基本類型作為參數傳遞時,傳遞的是這個值的拷貝。無論你怎麼改變這個拷貝,原值是不會改變的。
代碼如下:
1 public class ParameterTransfer { 2 public static void main(String[] args) { 3 int num = 30; 4 System.out.println("調用add方法前num=" + num); 5 add(num); 6 System.out.println("調用add方法後num=" + num); 7 } 8 9 public static void add(int param) { 10 param = 100; 11 } 12 }View Code
代碼的運行結果如下:
調用add方法前num=30
調用add方法後num=30
num的值在運行add()方法前後值並未改變,為什麼呢?
當執行了int num = 30;這句代碼後,程式在棧記憶體中開闢了一塊地址為AD8500的記憶體,裡邊放的值是30,記憶體模型如下圖:
執行到add()方法時,程式在棧記憶體中又開闢了一塊地址為AD8600的記憶體,將num的值30傳遞進來,此時這塊記憶體裡邊放的值是30,執行param = 100;後,AD8600中的值變成了100。記憶體模型如下圖:
引用類型作為參數傳遞
引用類型作為參數傳遞時,傳遞的是引用地址。所以在改變引用類型的值時,原值會改變。
代碼如下:
1 public class ParameterTransfer { 2 public static void main(String[] args) { 3 String[] array = new String[] {"huixin"}; 4 System.out.println("調用reset方法前array中的第0個元素的值是:" + array[0]); 5 reset(array); 6 System.out.println("調用reset方法後array中的第0個元素的值是:" + array[0]); 7 } 8 9 public static void reset(String[] param) { 10 param[0] = "hello, world!"; 11 } 12 13 }View Code
運行的結果如下:
調用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變數的值。原因可能有以下兩個:- 一個是String實際上操作的是char[],可以理解為String是char[]的包裝類。
- 二是給String變數重新賦值後,實際上沒有改變這個變數的值,而是重新new了一個String對象,改變了新對象的值,所以原來的String變數的值並沒有改變。