李 冉,陳蘭芹
(荊楚理工學院 計算機工程學院,湖北 荊門448000)
Java語言中,參數傳遞機制對于初學者來說,一直是一個難點,也是一個重點。關于這方面的教材、論文和參考手冊很多,但是說法不一,有的模糊帶過。比如,有的認為Java的參數傳遞分兩種情況,即值傳遞和引用傳遞;有的認為Java中所有的參數傳遞都是按值傳遞。其實這些說法都有一定的合理性,但是沒有清晰地說明它的基本原理,容易讓初學者陷入邏輯漩渦。本文參閱了各種說法,并結合了JDK的幫助文檔,首先從Java的變量內存分配機制開始,全面深入地剖析Java語言中參數傳遞機制,并得出最合理的結論。
Java的內存分配機制很復雜,這里只分析Java的變量內存分配規則,以助于理解Java的參數傳遞機制,而不考慮它在堆區、棧區還是靜態數據區,也不考慮生命周期。
Java的數據類型分為兩大類,分別是基本數據類型和引用數據類型[1]。基本數據類型,也稱為簡單數據類型,包括byte、char、short、int、long、float、double和boolean共8種;引用數據類型也稱為復雜數據類型,包括接口、類和數組共3種。
JVM根據變量的數據類型來分配內存空間,Java中不同大類型的變量內存分配規則不同。
1)對于基本數據類型
JVM對于基本數據類型的變量,只分配一定大小的內存單元,用于存儲該變量的值。如果復制該變量,則是復制該內存單元的值,例如:
double x=7.3;//定義了雙精度類型變量x,并賦初值為7.3
double y=x;//定義了雙精度類型變量y,并copy了x的值
對應的內存單元示意圖見圖1。

圖1 基本數據類型變量內存分配圖
x變量的值復制給了y變量,此時,x、y是兩個值相同的但是完全獨立的兩個內存單元。
2)對于引用數據類型
JVM對于引用數據類型的變量,也是分配一定大小的內存單元,用于存儲該變量所指向的對象,該變量的值為對象的引用。如果復制該變量,實際上是復制了該變量內存單元的值,而不是復制它所指的對象的值,例如:
Point p1=new Point(23,34);//Point為Java中一個已有的類
Point p2=p1;//將p1的值復制給了p2變量對應的內存單元示意圖見圖2。

圖2 引用數據類型變量內存分配圖
p1變量的值復制給了p2變量,此時p1與p2存放的值是同一個對象的引用,即指向同一個對象,但是它們是兩個完全獨立的內存單元。
對于p1和它所指向的對象,本質上也是兩個完全不同的內存單元。
在Java語言中,參數傳遞是在程序運行過程中,實際參數將參數值傳遞給被調用的方法中相應的形式參數,然后實現對數據處理,或者完成特定的功能。實際參數是調用過程中,從主調方法傳遞給被調用方法的值,它可以是常量、變量或者表達式;形式參數是被調用方法用于接收并存儲實際參數值的變量[2]。
Java語言中,參數傳遞的機制只有一個,即將實際參數的值復制一份賦值給形式參數,形式參數值的任何變化,不會影響到實參的值。實際參數可以是常量、變量或者表達式,變量又有很多種類型,但是都遵循這個機制。
Java語言中,常量和表達式都有值的屬性,在參數傳遞中,將常量的值或者表達式計算的值復制一份賦值給形式參數。在方法內部,形式參數的值發生任何改變,不會影響到實際參數的值。實際上,常量和表達式的值不允許修改,也不可能修改。
實際參數為基本類型變量時,參數傳遞過程中,將實參變量的值復制一份賦值給形式參數。實參變量和形參變量是兩個獨立的內存單元,形參變量的值發生任何變化,不會影響到實參變量的值,如下例程很好地驗證這種傳遞邏輯關系。


運行結果如下:
在fun方法中,形參d的初始值,d=12.3
在fun方法中,形參d的初始值,d=15.3
在main方法中,實參f在fun方法執行之后的值,f=12.3
執行過程如圖3所示。

圖3 實參為簡單類型變量的參數傳遞執行過程示意圖
實際參數為引用類型變量時,參數傳遞不是實參變量所指向的對象的值,而是實參變量本身的值,即所指對象的引用。傳遞的實現就是將實參變量所存儲的引用的值復制一份賦值給形參變量,形參變量的內存單元也存儲了該對象的引用值。但是形參變量與實參變量是兩個獨立的存儲單元,形參變量內存單元值的變化不會影響到實參變量。如下例程很好地驗證了這種邏輯關系。


運行結果如下:


圖4 實參為引用類型變量的參數傳遞執行過程示意圖
由圖3和圖4可以看出,引用類型實參到形參的參數傳遞,與簡單類型實參到形參的傳遞方式一樣,都是值的復制,只不過被復制的值的含義不同而已。
在Java語言中,參數傳遞的機制本質上就是值的復制,即只是從實際參數到形式參數變量的值的復制。也可以說,在Java中,參數傳遞都是按值傳遞的,就像快遞公司送包裹一樣,不關心包裹的內容是什么。一些文章或者書籍中,將參數傳遞分為按值傳遞和按引用傳遞,是考慮了參數值的意義,這與Java參數傳遞的實現機制是沒有關系的。初學者只要搞清楚了Java中變量類型分類和內存分配機制,就能靈活地使用參數傳遞機制。
[1]埃克爾.Java編程思想[M].陳昊鵬,譯.4版.北京:機械工業出版社,2007.
[2]昊斯特曼.Java核心技術:卷I[M].葉乃文,鄺勁筠,杜永萍,譯.北京:機械工業出版社,2008.