參考文章: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類型特殊,單獨說)
基本類型作為參數傳遞
基本類型作為參數傳遞時,傳遞的是這個值的拷貝。無論你怎麼改變這個拷貝,原值是不會改變的。
代碼如下:
![](http://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
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,記憶體模型如下圖:
![](http://images2015.cnblogs.com/blog/1110607/201702/1110607-20170222193024163-705619449.png)
執行到add()方法時,程式在棧記憶體中又開闢了一塊地址為AD8600的記憶體,將num的值30傳遞進來,此時這塊記憶體裡邊放的值是30,執行param = 100;後,AD8600中的值變成了100。記憶體模型如下圖:
引用類型作為參數傳遞
引用類型作為參數傳遞時,傳遞的是引用地址。所以在改變引用類型的值時,原值會改變。
代碼如下:
![](http://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
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的記憶體地址,如圖所示:![](http://images2015.cnblogs.com/blog/1110607/201702/1110607-20170222193612929-338293555.png)
![](http://images2015.cnblogs.com/blog/1110607/201702/1110607-20170222193656991-1607161727.png)
- 一個是String實際上操作的是char[],可以理解為String是char[]的包裝類。
- 二是給String變數重新賦值後,實際上沒有改變這個變數的值,而是重新new了一個String對象,改變了新對象的值,所以原來的String變數的值並沒有改變。