java中方法的參數傳遞機制

  問:當一個對象被看成參數傳遞到一個方法後,此方法可改變這個對象的屬性,並可返回變化後的結果,那麼這裏究竟是值傳遞仍是引用傳遞? 
  答:是值傳遞。Java 編程語言只有值傳遞參數。當一個對象實例做爲一個參數被傳遞到方法中時,參數的值就是該對象的引用一個副本。指向同一個對象,對象的內容能夠在被調用的方法中改變,但對象的引用(不是引用的副本)是永遠不會改變的。html

  Java參數,無論是原始類型仍是引用類型,傳遞的都是副本(有另一種說法是傳值,可是說傳副本更好理解吧,傳值一般是相對傳址而言)。java

  若是 參數類型是原始類型,那麼傳過來的就是這個參數的一個副本,也就是這個原始參數的值,這個跟以前所談的傳值是同樣的。若是在函數中改變了副本的值 不會改變原始的值.
  若是 參數類型是引用類型,那麼傳過來的就是這個引用參數的副本,這個副本存放的是參數的地址。若是在函數中沒有改變這個副本的地址,而是改變了地址中的 值,那麼在函數內的 改變會影響到傳入的參數。若是在函數中改變了副本的地址,如new一個,那麼副本就指向了一個新的地址,此時傳入的參數仍是指向原來的 地址,因此不會改變參數的值。
基本類型參數傳遞:不改變值
       
引用類型參數傳遞:改變值
       
 
 
 

不管是什麼語言,要討論參數傳遞方式,就得從內存模型提及,主要是我我的以爲從內存模型來講參數傳遞更爲直觀一些。閒言少敘,下面咱們就經過內存模型的方式來討論一下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

                    Java中的參數傳遞方式
  執行到add()方法時,程序在棧內存中又開闢了一塊地址爲AD8600的內存,將num的值30傳遞進來,此時這塊內存裏邊放的值是30,執行param = 100;後,AD8600中的值變成了100。內存模型以下圖:   
                 Java中的參數傳遞方式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!

當對象做爲參數傳遞時,傳遞的是對象的引用,也就是對象的地址。下邊用內存模型圖來講明。

       Java中的參數傳遞方式

  當程序執行了String[] array = new String[] {"huixin"}後,程序在棧內存中開闢了一塊地址編號爲AD9500內存空間,用於存放array[0]的引用地址,裏邊放的值是堆內存中的一個地址,示例中的值爲BE2500,能夠理解爲有一個指針指向了堆內存中的編號爲BE2500的地址。堆內存中編號爲BE2500的這個地址中存放的纔是array[0]的值:huixin。

  當程序進入reset方法後,將array的值,也就是對象的引用BE2500傳了進來。這時,程序在棧內存中又開闢了一塊編號爲AD9600的內存空間,裏邊放的值是傳遞過來的值,即AD9600。能夠理解爲棧內存中的編號爲AD9600的內存中有一個指針,也指向了堆內存中編號爲BE2500的內存地址,如圖所示:

       Java中的參數傳遞方式

  這樣一來,棧內存AD9500和AD9600(即array[0]和param的值)都指向了編號爲BE2500的堆內存。

  在reset方法中將param的值修改成hello, world!後,內存模型以下圖所示:

       Java中的參數傳遞方式

  改變對象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變量的值並無改變。

相關文章
相關標籤/搜索