關於隱藏和覆蓋的區別,要提到RTTI(run-time type identification)(運行期類型檢查),也就是運行期的多態,當一個父類引用指向子類對象的時候,請看下面我編寫的一段代碼:
代碼如下:
public class RunTime {
public static void main(String[] args) {
Animal a = new Cat();
System.out.println(a.A);
System.out.println(a.b);
a.voice();
a.method();
}
}
class Dog extends Animal {
public int b = 3;
public static int A = 3;
public static void method(){
System.out.println("狗");
}
public void voice() {
System.out.println("狗叫");
}
}
class Cat extends Animal {
public int b = 4;
public static int A = 4;
public static void method(){
System.out.println("貓");
}
public void voice() {
System.out.println("貓叫");
}
}
class Animal {
public int b = 0;
public static int A = 0;
public static void method(){
System.out.println("動物");
}
public void voice() {
System.out.println("動物叫");
}
}
輸出結果是:
0
0
貓叫
動物
您可以看到,當父類Animal的引用a指向子類Dog時,RTTI在運行期會自動確定該引用的真是類型,當子類 覆蓋 了父類的方法時,則直接調用子類的方法,打印出“貓叫”;然而非靜態的方法在子類中重寫的話就是被覆蓋,而靜態的方法被子類重寫的話就是隱藏,另外,靜態變量和成員變量也是被隱藏,而RTTI是只針對覆蓋,不針對影藏,所以,靜態變量 A 和 非靜態變量 b 以及靜態方法method() 均不通過RTTI,是哪個類的引用就調用誰的靜態方法,成員變量,而這裡是父類Animal的引用,所以直接調用父類Animal中的方法以及成員變量。所以靜態方法 method(), 靜態變量 A 和成員變量 b 打印結果全是父類中的。只用被覆蓋的非靜態方法voice()才打印子類的。