在不同的
java新聞組中,參數是傳值還是傳址一直是一個經常被爭辯的話題。誤解的中心是以下兩個事實:
對象是傳引用的
參數是傳值的
這兩個能夠同時成立嗎?一個字:是!在java中,你從來沒有傳遞對象,你傳遞的僅僅是對象的引用!一句話,java是傳引用的。然而,當你傳遞一個參數,那么只有一種參數傳遞
機制:傳值!
通常,當程序員討論傳值和傳引用時,他們是指語言的參數傳遞機制,c++同時支持這兩種機制,因此,以前使用過c++的
程序員開始好像不能確定的java是如何傳參數的。java語言為了事情變得簡單只支持參數傳值的機制。
java中的變量有兩種類型:引用類型和原始類型。當他們被作為參數傳遞給方法時,他們都是傳值的。這是一個非常重要的差別,下面的代碼范例將說明這一點。在繼續(xù)前,我們有必要定義一下傳值和傳引用。
傳值意味著當參數被傳遞給一個方法或者函數時,方法或者函數接收到的是原始值的副本。因此,如果方法或者函數修改了參數,受影響的只是副本,原始值保持不變。
關于java中的參數傳遞的混亂是因為很多java程序員是從c++轉變過來的。c++有引用和非引用類型的變量,并且分別是通過傳引用和傳值得到的。java語言有原始類型和對象引用,那么,按照
邏輯,java對于原始類型使用傳值而對引用類型是傳引用的,就像c++一樣。畢竟,你會想到如果你正在傳遞一個引用,那么它一定是傳引用的。這是一個很誘惑人的想法,但是是錯誤的!
在c++和java中,當函數的參數不是引用時,你傳遞的是值得副本(傳值)。但是對于引用類型就不同了。在c++中,當參數是引用類型,你傳遞的是引用或者內存地址(傳引用),而在java中,傳遞一個引用類型的參數的結果只是傳遞引用的副本(傳值)而非引用自身。
這是一個非常重要的區(qū)別!java不考慮參數的類型,一律傳遞參數的副本。仍然不信?如果java中是傳引用,那么下面的范例中的swap方法將交換他們的參數。因為是傳值,因此這個方法不是像期望的那樣正常
工作。
class Swap { public static void main(String args[]) { Integer a, b; int i,j; a = new Integer(10); b = new Integer(50); i = 5; j = 9; System.out.println(\"Before Swap, a is \" + a); System.out.println(\"Before Swap, b is \" + b); swap(a, b); System.out.println(\"After Swap a is \" + a); System.out.println(\"After Swap b is \" + b); System.out.println(\"Before Swap i is \" + i); System.out.println(\"Before Swap j is \" + j); swap(i,j); System.out.println(\"After Swap i is \" + i); System.out.println(\"After Swap j is \" + j); } public static void swap(Integer ia, Integer ib) { Integer temp = ia; ia = ib; ib = temp; } public static void swap(int li, int lj) { int temp = li; li = lj; lj = temp; } } |
上面程序的輸出是:
Before Swap, a is 10
Before Swap, b is 50
After Swap a is 10
After Swap b is 50
Before Swap i is 5
Before Swap j is 9
After Swap i is 5
After Swap j is 9
因為swap方法接收到的是引用參數的副本(傳值),對他們的修改不會反射到調用代碼。
譯者注:在傳遞引用和原始類型時還是有不同的,考慮以下的代碼:
class Change { public static void main(String args[]) { StringBuffer a=new StringBuffer(\"ok\"); int i; i = 5; System.out.println(\"Before change, a is \" + a); change(a); System.out.println(\"After change a is \" + a); System.out.println(\"Before change i is \" + i); change(i); System.out.println(\"After change i is \" + i); } public static void change(StringBuffer ia) { ia.append(\" ok?\"); } public static void change(int li) { li = 10; } }
|
程序的輸出為:
Before change, a is ok
After change a is ok ok?
Before change i is 5
After change i is 5
即如果
傳遞的是引用,那么可以修改引用對象的內容,這個改變會影響到原來的對象,而傳遞的如果是原始類型則不會有影響。這個也是造成誤解的原因之一吧。