看板 java 關於我們 聯絡資訊
這題是繼承和有沒使用到多型的問題。 有請打開Debugger,以Eclipse為例 在child c1 =new child();的下一行下斷點。 然後打開c1,你會發現有兩個i,一個是來自parent,一個是child本身。 什麼?你不相信我? 為證明我說的,我把程式稍微加料一下。 在parent p1 =new parent();的下一行加上c1.showSuper(); 然後在class child裡面加上 void showSuper() { System.out.println("show c1.super.i: "+super.i); } 這樣就應該了解了。 另外, 如果你有使用C/C++也可以把你原本的code改成C/C++的版本。 像是Visual Studio 2012 Desktop Expression是免費的, 打開Debugger來跑,你會看到child裡面還會有一個parent的分支可以打開。 這個Debugger算是做的比較清楚的。 最後, 雖然Java沒辦法操作指標,但物件指向相當於指標指向的概念。 那toString()是把指標的值印出來。 而C/C++有個%p可以把指標的值印出來 所以你在C/C++下 1 cout << "address of p = " << &p << endl; 2 cout << "address of c = " << &c << endl; 3 printf("value of p = %p\n", p); 4 printf("value of c = %p\n", c); 列1, 2永遠不會變的,就是指標本身在記憶體裡面的位址。 列3, 4在p轉向c之後,兩個值會一樣。就是toString()所呈現的。 ※ 引述《orze04 (orz)》之銘言: : public class QQ { : public static void main(String[] args){ : child c1 =new child(); : parent p1 =new parent(); : System.out.println(p1.toString()+" "+c1.toString()); : System.out.println("p1.i: "+p1.i); : p1.show(); : System.out.println("c1.i: "+c1.i); : c1.show(); : System.out.println(""); : p1=c1; : System.out.println(p1.toString()+" "+c1.toString()); : System.out.println("p1.i: "+p1.i); : p1.show(); : System.out.println("c1.i: "+c1.i); : c1.show(); : System.out.println(""); : } : } : class parent{ : int i; : parent(){} : void show(){ : System.out.println("show p1.i: "+i); : } : } : class child extends parent{ : int i=1; : child(){} : void show(){ : System.out.println("show c1.i: "+i); : } : } : result: : parent@14a55f2 child@15093f1 : p1.i: 0 : show p1.i: 0 : c1.i: 1 : show c1.i: 1 : child@15093f1 child@15093f1 : p1.i: 0 <<~~!? : show c1.i: 1 : c1.i: 1 : show c1.i: 1 : 在執行p1=c1後 : 透過hashCode可以看到p1和c1已經變成同樣內容 : 為何要顯示p.i時還是指向parent的class? : 還是說p1=c1只有methodc會被蓋掉,本身成員變數不會? -- ※ 發信站: 批踢踢實業坊(ptt.cc) ◆ From: 114.43.112.210
orze04:我的意思是為何"p1.i"還會指向原本p1.i的記憶體位置? 03/26 19:17
bleed1979:c1包含有parent的i,當p1指向c1時取i是取c1的parent的i 03/26 19:20
orze04:噢噢 這樣我就懂了 03/26 19:39
mars90226:簡單來說,看型別決定用哪個i 03/26 19:55
dahanhsi:那為什麼用show()的時候會拿到1呢? 03/26 20:09
bleed1979:一開始有講是多型的問題。。。 03/26 20:17
ssccg:這不止是多型問題,而是java預設在child class有同名的時候 03/27 02:03
ssccg:method就是override而不是shadow 03/27 02:06
ssccg:但像C++、C#是有virtual的method才是override,否則是shadow 03/27 02:07
ssccg:而成員變數在各語言都是shadow,因為直接存取變數已經連封裝 03/27 02:12
ssccg:都沒了,如果要多型要對getter/setter做 03/27 02:14
bleed1979:只有parent有getI()但child僅繼承getI()的情況 03/27 02:24
bleed1979:沒method override就仍會是parent的。 03/27 02:33
bleed1979:應該不是"不止",而是"就是"多型效果有沒有出來的問題。 03/27 02:44