java經(jīng)典問題:傳值還是傳引用(轉(zhuǎn))
java經(jīng)典問題:傳值還是傳引用(轉(zhuǎn))[@more@]經(jīng)典的問題,但卻不容易弄懂,尤其對(duì)有c基礎(chǔ)的java程序員來說,更容易引起混亂,這里我試圖簡單點(diǎn)描述。
“java函數(shù)是傳值的,java函數(shù)傳遞的參數(shù)是對(duì)象的引用”
這兩句話好像初聽上去有些矛盾,但卻是事實(shí),因而引起很多初學(xué)者的混亂。在這里我試圖據(jù)個(gè)簡單的例子來說明java的這個(gè)特性,可能不全面,希望大家來補(bǔ)全。
public class TestRef {
public static void main(String[] args)
{
ValueObject vo1 = new ValueObject("A", 1);
System.out.println("after vo1: " + vo1.getName()); //=A
changeValue1(vo1);
System.out.println("after changeValue1: " + vo1.getName());
//=A1, changed
changeValue2(vo1);
System.out.println("after changeValue2: " + vo1.getName());
//=A1, changeValue2內(nèi)部的賦值不會(huì)影響這里。
}
/**
* 使用vo1自身的函數(shù)對(duì)其內(nèi)部數(shù)據(jù)進(jìn)行改變是有效的,函數(shù)外可反映出來
* 這種object稱為可變的(mutable)
* @param vo1
*/
private static void changeValue1(ValueObject vo1) {
vo1.setName("A1");
}
/**
* 在函數(shù)內(nèi)給vo1重新賦值不會(huì)改變函數(shù)外的原始值
* @param vo1
*/
private static void changeValue2(ValueObject vo1) {
vo1 = new ValueObject("B", 2);
System.out.println("inside changeValue2: "+ vo1.getName());
//=B,賦值操作引起的結(jié)果變化僅在changeValue2內(nèi)部有效
}
}
class ValueObject {
public ValueObject() {}
public ValueObject(String name, int id)
{
this.name = name;
this.id = id;
}
private String name;
private int id;
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
解釋,vo1作為一個(gè)object,當(dāng)它被用作函數(shù)參數(shù)的時(shí)候傳遞給函數(shù)的是一個(gè)引用值,這個(gè)名稱有點(diǎn)怪,又有引用又有值,到底是引用還是值呢,就看你怎么理解了。如果你是去考試,官方的答案是值??墒强雌饋碛窒笠冒?,希望從這個(gè)例子,你能理解java參數(shù)傳遞和和C/C++程序中的引用傳遞的不同的地方。另外,這也是java作為OO語言的特性之一:封裝的體現(xiàn)。
先講一下對(duì)象賦值的關(guān)系,舉例來說,下列代碼:
ValueObject v2, v3;
v2 = new ValueObject("C", 3); 粗體的部分創(chuàng)建了一個(gè)數(shù)據(jù)結(jié)構(gòu),假設(shè)存放在內(nèi)存地址A000,賦值給句柄 v2
v3 = new ValueObject("D", 4); 粗體的部分創(chuàng)建了一個(gè)數(shù)據(jù)結(jié)構(gòu),假設(shè)存放在內(nèi)存地址B000,賦值給句柄 v3
v2 = v3; 這句話的作用是把操作B000的地址的句柄的值付給了v2的句柄,使得v2和v3一樣操作B000的地址,這意味著:
1.原來v2指向的地址A000變成無主的內(nèi)存地址,將自動(dòng)被jvm回收。
2.既然v2和v3指向同一片地址,對(duì)v3的修改v2也能得到,反之亦然。
整理得下列代碼,請(qǐng)感興趣的朋友運(yùn)行驗(yàn)證
ValueObject v2 = new ValueObject("C", 3);
ValueObject v3 = new ValueObject("D", 4);
v2 = v3;
System.out.println("after v2=v3");
System.out.println("v2= "+ v2.getName());//=D
System.out.println("v3= "+ v3.getName());//=D
v3.setName("C1");
System.out.println("after v3 setnameTo C1");
System.out.println("vo2= "+ v2.getName());//=C1
System.out.println("vo3= "+ v3.getName());//=C1
因此,可以得出結(jié)論,java中對(duì)象的每個(gè)實(shí)例(instance, 比如vo1, v2, v3 都是ValueObject的實(shí)例)的內(nèi)存地址是唯一的,它一旦被創(chuàng)建,能夠?qū)@個(gè)地址進(jìn)行操作的就是每個(gè)實(shí)例自己,如果ValueObject類中沒有public void setName之類的方法對(duì)這個(gè)類的實(shí)例中的數(shù)據(jù)進(jìn)行修改的話,程序是沒有任何別的方法可以修改ValueObject類的實(shí)例中的數(shù)據(jù),這個(gè)就是java的封裝特性。對(duì)于不提供修改內(nèi)部數(shù)據(jù)的方法的類,我們稱為不可變(immutable)的類。在函數(shù)中對(duì)傳入的參數(shù)變量進(jìn)行賦值操作,只能在函數(shù)范圍內(nèi)改變局部變量指向的引用地址,但是不會(huì)改變原始地址的內(nèi)容。因此,在changeValue2(...)函數(shù)內(nèi)部的vo1和函數(shù)外的vo1雖然名字相同,但是實(shí)際上是不同的實(shí)例變量,只不過指向了和函數(shù)外的vo1同樣的地址,所以當(dāng)我們用vo1=... 對(duì)其進(jìn)行賦值的時(shí)候,只不過是把函數(shù)內(nèi)的臨時(shí)變量指向了新的地址,并沒有改變原始vo1內(nèi)存地址中的內(nèi)容。這就是在運(yùn)行changeValue2(...)之后,vo1的值在main范圍內(nèi)仍然沒有被修改的原因。而changeValue1里面是調(diào)用的ValueObject本身的function來更改其內(nèi)容,因此是原始內(nèi)存地址中的數(shù)據(jù)被更改了,所以是全局有效的。