在不同的java新聞組中,參數是傳值還是傳址一直是一個經常被爭辯的話題。誤解的中心是以下兩個事實:
對象是傳引用的
參數是傳值的
這兩個能夠同時成立嗎?一個字:是!在java中,你從來沒有傳遞對象,你傳遞的僅僅是對象的引用!一句話,java是傳引用的。然而,當你傳遞一個參數,那麼只有一種參數傳遞機制:傳值!
通常,當程序員討論傳值和傳引用時,他們是指語言的參數傳遞機制,c++同時支持這兩種機制,因此,以前使用過c++的程序員開始好像不能確定的java是如何傳參數的。java語言為了事情變得簡單只支持參數傳值的機制。
java中的變量有兩種類型:引用類型和原始類型。當他們被作為參數傳遞給方法時,他們都是傳值的。這是一個非常重要的差別,下面的代碼范例將說明這一點。在繼續前,我們有必要定義一下傳值和傳引用。
傳值意味著當參數被傳遞給一個方法或者函數時,方法或者函數接收到的是原始值的副本。因此,如果方法或者函數修改了參數,受影響的只是副本,原始值保持不變。
關於java中的參數傳遞的混亂是因為很多java程序員是從c++轉變過來的。c++有引用和非引用類型的變量,並且分別是通過傳引用和傳值得到的。java語言有原始類型和對象引用,那麼,按照邏輯,java對於原始類型使用傳值而對引用類型是傳引用的,就像c++一樣。畢竟,你會想到如果你正在傳遞一個引用,那麼它一定是傳引用的。這是一個很誘惑人的想法,但是是錯誤的!
在c++和java中,當函數的參數不是引用時,你傳遞的是值得副本(傳值)。但是對於引用類型就不同了。在c++中,當參數是引用類型,你傳遞的是引用或者內存地址(傳引用),而在java中,傳遞一個引用類型的參數的結果只是傳遞引用的副本(傳值)而非引用自身。
這是一個非常重要的區別!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
即如果傳遞的是引用,那麼可以修改引用對象的內容,這個改變會影響到原來的對象,而傳遞的如果是原始類型則不會有影響。這個也是造成誤解的原因之一吧。