對(duì)象的引用問(wèn)題(1)
@Test
public void MethodTest() {
Person person = new Person();
person.name = "hello";
show(person);
System.out.println("-"+person.name);
}
public void show(Person person) {
Person innerPerson = person;
System.out.println(innerPerson.name);
innerPerson.name = "賦值";
}
打印
hello
-賦值
結(jié)論 方法內(nèi) innerPerson 和 person 引用的是同一個(gè)地址空間 , innerPerson的賦值會(huì)對(duì)方法外的對(duì)象造成改變; 由于 innerPerson 是局部對(duì)象 存取速度也會(huì)比成員快;
如果把變量賦值為final 會(huì)保證他只能賦值一次 , 更安全
對(duì)象的引用問(wèn)題(2)
今天我在學(xué)習(xí)hanler 復(fù)用message sPool的時(shí)候 發(fā)現(xiàn)類型這樣的代碼
next 賦值為null , 不會(huì)造成 sPool 也為null嗎?
sPool = person.next;//next 是一個(gè)bean對(duì)象
person.next = null;
System.out.println(sPool);
打印
com.example.liuyuzhe.kotlindome.ObjectReferenceTest$User@694f9431
結(jié)果是有值的 , 經(jīng)過(guò)分析 , sPool = person.next 他們2個(gè)對(duì)象指向了同一個(gè)內(nèi)存地址空間 , person.next = null; 這句話是把 next 對(duì)象 的引用指向了另一個(gè)(地址空間)對(duì)象 (它是null對(duì)象) ; 你可以用 person.next 來(lái)改變sPool對(duì)象的內(nèi)部結(jié)構(gòu) , 但無(wú)法改他的地址空間
String 設(shè)計(jì)為不可以變類型是為了 安全避免 別名(賦值會(huì)改變: 別名定義在下面)的問(wèn)題
private boolean mChange = false;
private boolean tempBoolean = false;
public void String_isChange() throws Exception {
new Thread() {
@Override
public void run() {
for (; ; ) {
fetchSleep(200);
if (isChange()) {
varStr = " start change ";
tempBoolean = true;
System.out.println("start change" + varStr);
fetchSleep(5000);
break;
}
}
}
}.start();
varStr = "init";
testVar(varStr);
}
private void fetchSleep(int millis) {
try {
Thread.sleep(millis);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
/**
* 11-05 23:50:34.324 2920-2920/com.example.liuyuzhe.kotlindome I/System.out: tempBoolean start false
11-05 23:50:34.525 2920-2953/com.example.liuyuzhe.kotlindome I/System.out: start change start change
11-05 23:50:37.326 2920-2920/com.example.liuyuzhe.kotlindome I/System.out: tempBoolean end true
11-05 23:50:37.326 2920-2920/com.example.liuyuzhe.kotlindome I/System.out: fetchSleep after init
* @param varStr
*/
public void testVar(String varStr) {
mChange = true;
System.out.println(" fetchSleep before " + varStr);
System.out.println("tempBoolean start " + tempBoolean);
fetchSleep(3000);
//走到這里 tempBoolean 打印為true , 說(shuō)明 varStr的成員變量已經(jīng)改變了,
// 但是varStr 傳遞進(jìn)來(lái)時(shí)其值已經(jīng)確定了 , 不可變類型 , 成員變量的varStr 和 形參不是一個(gè)對(duì)象;
System.out.println("tempBoolean end " + tempBoolean);
System.out.println(" fetchSleep after " + varStr);
}
public boolean isChange() {
return mChange;
}
什么是別名 : 就是對(duì)象的引用地址 , 會(huì)由一個(gè)對(duì)象的改變 , 導(dǎo)致同一個(gè)引用的對(duì)象發(fā)生改變
public static void main(String[] args) {
// TODO Auto-generated method stub
Tank t1=new Tank();
Tank t2=new Tank();
t1.level=9;
t2.level=47;
System.out.println("t1.level:"+t1.level+" t2.level:"+t2.level);
t1=t2;
System.out.println("t1.level:"+t1.level+" t2.level:"+t2.level);
t1.level=27;
System.out.println("t1.level:"+t1.level+" t2.level:"+t2.level);
}
}
結(jié)果:
t1.level:9 t2.level:47
t1.level:47 t2.level:47
t1.level:27 t2.level:27
String 是不可變類型 , 不存在別名問(wèn)題
String t1 = "a";
String t2 = "b";
System.out.println("初始化 t1="+t1);
System.out.println("初始化 t2="+t2);
t2 = t1;
System.out.println("拉手后 t1="+t1);
System.out.println("拉手后 t2="+t2);
t1 = "c";
System.out.println("改變 t1 后 "+t1);
System.out.println("改變t1 后 t2的狀態(tài) "+t2);
打印結(jié)果
初始化 t1=a
初始化 t2=b
拉手后 t1=a
拉手后 t2=a
改變 t1 后 c
改變t1 后 t2的狀態(tài) a
說(shuō)明 String 的 值對(duì)象 , 不存在別名的問(wèn)題