類似的問題,又被人問到了幾次,決定還是簡單總結(jié)一下吧。這個問題,一些老手已經(jīng)很清楚了,但有時也會很不小心的被XX了一把。
其實問題的核心,就是參數(shù)雖然是傳的引用,但參數(shù)就是參數(shù),他自身是一個本地的局部引用而已,只不過在這個時刻和調(diào)用者指向了同一個對象。但并不代表這個局部引用在整個方法調(diào)用期間內(nèi)能始終和調(diào)用者保持一致。
下面是2個測試,分別測試可修改的Object和不可修改的
/**
* JAVA里面對象參數(shù)的陷阱
*/
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() {
// 當(dāng)然是15
System.out.println(v.i);
// 第一次調(diào)用
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) {
// 重新設(shè)置一個對象
v = new Value();
// 此時這里的v也是一個局部變量
// 但和類屬性的v已經(jīng)不相等了
// 修改這個v指向?qū)ο蟮臄?shù)值,已經(jīng)不影響類里面的屬性v了。
System.out.println(v == this.v);
v.i = 25;
}
}
class TestInteger {
// 初始化
Integer v = new Integer(15);
public void first() {
// 當(dāng)然是15
System.out.println(v);
// 第一次調(diào)用
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;
// 當(dāng)然,他們也不再相等
System.out.println(v == this.v);
}
public void third(Integer v) {
// 重新設(shè)置一個對象
v = new Integer(25);
// 此時這里的v也是一個局部變量
// 但和類屬性的v已經(jīng)不相等了
// 修改這個v指向?qū)ο蟮臄?shù)值,已經(jīng)不影響類里面的屬性v了。
System.out.println(v == this.v);
}
}
運行結(jié)果
15
true
20
false
20
15
true
false
15
false
15
希望這個例子能解開一些初學(xué)者的疑問。
相關(guān)推薦:全國計算機等級考試將于3月28日至4月1日舉行北京 | 天津 | 上海 | 江蘇 | 山東 |
安徽 | 浙江 | 江西 | 福建 | 深圳 |
廣東 | 河北 | 湖南 | 廣西 | 河南 |
海南 | 湖北 | 四川 | 重慶 | 云南 |
貴州 | 西藏 | 新疆 | 陜西 | 山西 |
寧夏 | 甘肅 | 青海 | 遼寧 | 吉林 |
黑龍江 | 內(nèi)蒙古 |