在不同的
java新聞組中,參數(shù)是傳值還是傳址一直是一個(gè)經(jīng)常被爭(zhēng)辯的話題。誤解的中心是以下兩個(gè)事實(shí):
對(duì)象是傳引用的
參數(shù)是傳值的
這兩個(gè)能夠同時(shí)成立嗎?一個(gè)字:是!在java中,你從來(lái)沒(méi)有傳遞對(duì)象,你傳遞的僅僅是對(duì)象的引用!一句話,java是傳引用的。然而,當(dāng)你傳遞一個(gè)參數(shù),那么只有一種參數(shù)傳遞
機(jī)制:傳值!
通常,當(dāng)程序員討論傳值和傳引用時(shí),他們是指語(yǔ)言的參數(shù)傳遞機(jī)制,c++同時(shí)支持這兩種機(jī)制,因此,以前使用過(guò)c++的
程序員開始好像不能確定的java是如何傳參數(shù)的。java語(yǔ)言為了事情變得簡(jiǎn)單只支持參數(shù)傳值的機(jī)制。
java中的變量有兩種類型:引用類型和原始類型。當(dāng)他們被作為參數(shù)傳遞給方法時(shí),他們都是傳值的。這是一個(gè)非常重要的差別,下面的代碼范例將說(shuō)明這一點(diǎn)。在繼續(xù)前,我們有必要定義一下傳值和傳引用。
傳值意味著當(dāng)參數(shù)被傳遞給一個(gè)方法或者函數(shù)時(shí),方法或者函數(shù)接收到的是原始值的副本。因此,如果方法或者函數(shù)修改了參數(shù),受影響的只是副本,原始值保持不變。
關(guān)于java中的參數(shù)傳遞的混亂是因?yàn)楹芏鄇ava程序員是從c++轉(zhuǎn)變過(guò)來(lái)的。c++有引用和非引用類型的變量,并且分別是通過(guò)傳引用和傳值得到的。java語(yǔ)言有原始類型和對(duì)象引用,那么,按照
邏輯,java對(duì)于原始類型使用傳值而對(duì)引用類型是傳引用的,就像c++一樣。畢竟,你會(huì)想到如果你正在傳遞一個(gè)引用,那么它一定是傳引用的。這是一個(gè)很誘惑人的想法,但是是錯(cuò)誤的!
在c++和java中,當(dāng)函數(shù)的參數(shù)不是引用時(shí),你傳遞的是值得副本(傳值)。但是對(duì)于引用類型就不同了。在c++中,當(dāng)參數(shù)是引用類型,你傳遞的是引用或者內(nèi)存地址(傳引用),而在java中,傳遞一個(gè)引用類型的參數(shù)的結(jié)果只是傳遞引用的副本(傳值)而非引用自身。
這是一個(gè)非常重要的區(qū)別!java不考慮參數(shù)的類型,一律傳遞參數(shù)的副本。仍然不信?如果java中是傳引用,那么下面的范例中的swap方法將交換他們的參數(shù)。因?yàn)槭莻髦?,因此這個(gè)方法不是像期望的那樣正常
工作。
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
因?yàn)閟wap方法接收到的是引用參數(shù)的副本(傳值),對(duì)他們的修改不會(huì)反射到調(diào)用代碼。
譯者注:在傳遞引用和原始類型時(shí)還是有不同的,考慮以下的代碼:
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
即如果
傳遞的是引用,那么可以修改引用對(duì)象的內(nèi)容,這個(gè)改變會(huì)影響到原來(lái)的對(duì)象,而傳遞的如果是原始類型則不會(huì)有影響。這個(gè)也是造成誤解的原因之一吧。