類似的問題,又被人問到了幾次,決定還是簡單總結一下吧。這個問題,一些老手已經很清楚了,但有時也會很不小心的被XX了一把。
其實問題的核心,就是參數雖然是傳的引用,但參數就是參數,他自身是一個本地的局部引用而已,只不過在這個時刻和調用者指向了同一個對象。但並不代表這個局部引用在整個方法調用期間內能始終和調用者保持一致。
下面是2個測試,分別測試可修改的Object和不可修改的
/**
* Java裡面對象參數的陷阱
*
* @author 老紫竹的家(laozizhu.com)
*
*/
public class Test {
public static void main(String[] args) {
TestValue tv = new TestValue();
tv.first();
TestInteger ti = new TestInteger();
ti.first();
}
}
class TestValue {
class Value {
public int i = 15;
}
// 初始化
Value v = new Value();
public void first() {
// 當然是15
System.out.println(v.i);
// 第一次調用
second(v);
System.out.println(v.i);
third(v);
System.out.println(v.i);
}
public void second(Value v) {
// 此時這裡的v是一個局部變量
// 和類屬性的v相等
System.out.println(v == this.v);
v.i = 20;
}
public void third(Value v) {
// 重新設置一個對象
v = new Value();
// 此時這裡的v也是一個局部變量
// 但和類屬性的v已經不相等了
// 修改這個v指向對象的數值,已經不影響類裡面的屬性v了。
System.out.println(v == this.v);
v.i = 25;
}
}
class TestInteger {
// 初始化
Integer v = new Integer(15);
public void first() {
// 當然是15
System.out.println(v);
// 第一次調用
second(v);
System.out.println(v);
third(v);
System.out.println(v);
}
public void second(Integer v) {
// 此時這裡的v是一個局部變量
// 和類屬性的v相等
System.out.println(v == this.v);
// 但這一句和前面的不同,雖然也是給引用賦值,但因為Integer是不可修改的
// 所以這裡會生成一個新的對象。
v = 20;
// 當然,他們也不再相等
System.out.println(v == this.v);
}
public void third(Integer v) {
// 重新設置一個對象
v = new Integer(25);
// 此時這裡的v也是一個局部變量
// 但和類屬性的v已經不相等了
// 修改這個v指向對象的數值,已經不影響類裡面的屬性v了。
System.out.println(v == this.v);
}
}
運行結果
15
true
20
false
20
15
true
false
15
false
15
希望這個例子能解開一些初學者的疑問。