extends 是繼承父類,只要那個類不是聲明final或者定義為abstract就能繼承,JAVA中不支持多重繼承,繼承只能繼承一個類,但implements可以實現多個接口,用逗號分開就行了。
比如:
class A extends B implements C,D,E(){ //class子類名extends父類名implements接口名
}
父類與子類繼承關系上的不同:
class A{
int i;
void f(){
}
}
class B extends A{
int j;
void f(){} //重寫方法
void g(){}
}
B b=new B();
b就是子類對象的實例,不僅能夠方位自己的屬性和方法,也能訪問父類的屬性和方法。諸如b.i,b.j.b.fn()都是合法的。此時b.f()是B中的f()
A a=new B();
a雖然是用的B的構造函數,但經過upcast,成為父類對象的實例,不能訪問子類的屬性和方法。a.i,a.f()是合法的,而a.j,a.g()是非法的。此時訪問a.f()是訪問B中的f();
實現過程
A a = new B(); 這條語句,實際上有三個過程:
(1) A a;
將a聲明為父類對象,只是一個引用,未分配空間
(2) B temp = new B();
通過B類的構造函數建立了一個B類對象的實例,也就是初始化
(3) a = (A)temp;
將子類對象temp轉換未父類對象并賦給a,這就是上傳(upcast),是安全的。
經過以上3個過程,a就徹底成為了一個A類的實例。
子類往往比父類有更多的屬性和方法,上傳只是舍棄,是安全的;而下傳(downcast)有時會增加,通常是不安全的。
多態
a.f()對應的應該是B類的方法f()
調用構造函數建立實例后,對應方法的入口已經確定了。
如此一來,a雖被上傳為A類,但其中重寫的方法f()仍然是B的方法()。也就是說,每個對象知道自己應該調用哪個方法。
A a1 = new B();
A a2 = new C();
a1,a2兩個雖然都是A類對象,但各自的f()不同。這正是多態性的體現。