java 繼承
繼承是使用現(xiàn)有類的所有功能,并在無需重新編寫原來的類的情況下對這些功能進(jìn)行擴展。
通過繼承創(chuàng)建的新類稱為“子類”,被繼承的類稱為“父類”。
java 的繼承就是子類繼承父類的特征和行為,使得子類對象具有父類的范例域和方法,或子類從父類繼承方法,使得子類具有父類相同的行為。
1. java 類繼承的語法
在 java 中通過 extends 關(guān)鍵字可以申明一個類是從另外一個類繼承而來:
class parentclass { } class childclass extends parentclass { }
2. java 類繼承的案例
我們要開發(fā)動物類,其中動物分別為企鵝以及老鼠,要求如下:
- 企鵝:屬性(姓名,id),方法(吃,睡,自我介紹)
- 老鼠:屬性(姓名,id),方法(吃,睡,自我介紹)
企鵝類:
public class penguin { private string name; private int id; public penguin(string myname, int myid) { name = myname; id = myid; } public void eat(){ system.out.println(name+"正在吃"); } public void sleep(){ system.out.println(name+"正在睡"); } public void introduction() { system.out.println("大家好!我是" + id + "號" + name + "."); } }
老鼠類:
public class mouse { private string name; private int id; public mouse(string myname, int myid) { name = myname; id = myid; } public void eat(){ system.out.println(name+"正在吃"); } public void sleep(){ system.out.println(name+"正在睡"); } public void introduction() { system.out.println("大家好!我是" + id + "號" + name + "."); } }
從這兩段代碼可以看出來,代碼存在重復(fù)了,導(dǎo)致后果就是代碼量大且臃腫,而且維護(hù)性不高(維護(hù)性主要是后期需要修改的時候,就需要修改很多的代碼,容易出錯),所以要從根本上解決這兩段代碼的問題,就需要繼承,將兩段代碼中相同的部分提取出來組成 一個父類:
公共父類:
public class animal { private string name; private int id; public animal(string myname, int myid) { name = myname; id = myid; } public void eat(){ system.out.println(name+"正在吃"); } public void sleep(){ system.out.println(name+"正在睡"); } public void introduction() { system.out.println("大家好!我是" + id + "號" + name + "."); } }
這個animal類就可以作為一個父類,然后企鵝類和老鼠類繼承這個類之后,就具有父類當(dāng)中的屬性和方法,子類就不會存在重復(fù)的代碼,維護(hù)性也提高,代碼也更加簡潔,提高代碼的復(fù)用性(復(fù)用性主要是可以多次使用,不用再多次寫同樣的代碼) 繼承之后的代碼:
企鵝類:
public class penguin extends animal { public penguin(string myname, int myid) { super(myname, myid); } }
老鼠類:
public class mouse extends animal { public mouse(string myname, int myid) { super(myname, myid); } }
3. 繼承類型
繼承的特性
子類擁有父類非 private 的屬性、方法。
子類可以擁有自己的屬性和方法,即子類可以對父類進(jìn)行擴展。
子類可以用自己的方式實現(xiàn)父類的方法。
java 的繼承是單繼承,但是可以多重繼承,單繼承就是一個子類只能繼承一個父類,多重繼承就是,例如 b 類繼承 a 類,c 類繼承 b 類,所以按照關(guān)系就是 b 類是 c 類的父類,a 類是 b 類的父類,這是 java 繼承區(qū)別于 c++ 繼承的一個特性。
提高了類之間的耦合性(繼承的缺點,耦合度高就會造成代碼之間的聯(lián)系越緊密,代碼獨立性越差)。
4. 繼承關(guān)鍵字
繼承可以使用 extends 和 implements 這兩個關(guān)鍵字來實現(xiàn)繼承,而且所有的類都是繼承于 java.lang.object,當(dāng)一個類沒有繼承的兩個關(guān)鍵字,則默認(rèn)繼承object(這個類在 java.lang 包中,所以不需要 import)祖先類。
1)extends關(guān)鍵字
在 java 中,類的繼承是單一繼承,也就是說,一個子類只能擁有一個父類,所以 extends 只能繼承一個類。
public class animal { private string name; private int id; public animal(string myname, string myid) { //初始化屬性值 } public void eat() { //吃東西方法的具體實現(xiàn) } public void sleep() { //睡覺方法的具體實現(xiàn) } } public class penguin extends animal{ }
2)implements關(guān)鍵字
使用 implements 關(guān)鍵字可以變相的使java具有多繼承的特性,使用范圍為類繼承接口的情況,可以同時繼承多個接口(接口跟接口之間采用逗號分隔)。
public interface a { public void eat(); public void sleep(); } public interface b { public void show(); } public class c implements a,b { }
3)super 與 this 關(guān)鍵字
super關(guān)鍵字:我們可以通過super關(guān)鍵字來實現(xiàn)對父類成員的訪問,用來引用當(dāng)前對象的父類。
this關(guān)鍵字:指向自己的引用。
class animal { void eat() { system.out.println("animal : eat"); } } class dog extends animal { void eat() { system.out.println("dog : eat"); } void eattest() { this.eat(); // this 調(diào)用自己的方法 super.eat(); // super 調(diào)用父類方法 } } public class test { public static void main(string[] args) { animal a = new animal(); a.eat(); dog d = new dog(); d.eattest(); } }
輸出結(jié)果為:
animal : eat dog : eat animal : eat
4)final關(guān)鍵字
final 關(guān)鍵字聲明類可以把類定義為不能繼承的,即最終類;或者用于修飾方法,該方法不能被子類重寫:
聲明類:
final class 類名 {//類體}
-
聲明方法:
修飾符(public/private/default/protected) final 返回值類型 方法名(){//方法體}
注:范例變量也可以被定義為 final,被定義為 final 的變量不能被修改。被聲明為 final 類的方法自動地聲明為 final,但是范例變量并不是 final
5. 構(gòu)造器
子類是不繼承父類的構(gòu)造器(構(gòu)造方法或者構(gòu)造函數(shù))的,它只是調(diào)用(隱式或顯式)。如果父類的構(gòu)造器帶有參數(shù),則必須在子類的構(gòu)造器中顯式地通過 super 關(guān)鍵字調(diào)用父類的構(gòu)造器并配以適當(dāng)?shù)膮?shù)列表。
如果父類構(gòu)造器沒有參數(shù),則在子類的構(gòu)造器中不需要使用 super 關(guān)鍵字調(diào)用父類構(gòu)造器,系統(tǒng)會自動調(diào)用父類的無參構(gòu)造器。
class superclass { private int n; superclass(){ system.out.println("superclass()"); } superclass(int n) { system.out.println("superclass(int n)"); this.n = n; } } // subclass 類繼承 class subclass extends superclass{ private int n; subclass(){ // 自動調(diào)用父類的無參數(shù)構(gòu)造器 system.out.println("subclass"); } public subclass(int n){ super(300); // 調(diào)用父類中帶有參數(shù)的構(gòu)造器 system.out.println("subclass(int n):"+n); this.n = n; } } // subclass2 類繼承 class subclass2 extends superclass{ private int n; subclass2(){ super(300); // 調(diào)用父類中帶有參數(shù)的構(gòu)造器 system.out.println("subclass2"); } public subclass2(int n){ // 自動調(diào)用父類的無參數(shù)構(gòu)造器 system.out.println("subclass2(int n):"+n); this.n = n; } } public class testsupersub{ public static void main (string args[]){ system.out.println("------subclass 類繼承------"); subclass sc1 = new subclass(); subclass sc2 = new subclass(100); system.out.println("------subclass2 類繼承------"); subclass2 sc3 = new subclass2(); subclass2 sc4 = new subclass2(200); } }
輸出結(jié)果為:
------subclass 類繼承------ superclass() subclass superclass(int n) subclass(int n):100 ------subclass2 類繼承------ superclass(int n) subclass2 superclass() subclass2(int n):200