繼承是java面向?qū)ο缶幊碳夹g(shù)的一塊基石卧檐,因為它允許創(chuàng)建分等級層次的類切蟋。
繼承就是子類繼承父類的特征和行為,使得子類對象(實例)具有父類的實例域和方法滚停,或子類從父類繼承方法沃粗,使得子類具有父類相同的行為。
生活中的繼承:
兔子和羊?qū)儆谑巢輨游镱惣耄{子和豹屬于食肉動物類最盅。
食草動物和食肉動物又是屬于動物類。
所以繼承需要符合的關(guān)系是:is-a起惕,父類更通用檩禾,子類更具體。
雖然食草動物和食肉動物都是屬于動物疤祭,但是兩者的屬性和行為上有差別盼产,所以子類會具有父類的一般特性也會具有自身的特性。
類的繼承格式
在 Java 中通過 extends 關(guān)鍵字可以申明一個類是從另外一個類繼承而來的勺馆,一般形式如下:
類的繼承格式
class 父類 {
}
class 子類 extends 父類 {
}
為什么需要繼承
接下來我們通過實例來說明這個需求戏售。
開發(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); } }
繼承類型
需要注意的是 Java 不支持多繼承心傀,但支持多重繼承屈暗。
繼承的特性
子類擁有父類非 private 的屬性、方法脂男。
子類可以擁有自己的屬性和方法养叛,即子類可以對父類進(jìn)行擴(kuò)展。
子類可以用自己的方式實現(xiàn)父類的方法宰翅。
Java 的繼承是單繼承弃甥,但是可以多重繼承,單繼承就是一個子類只能繼承一個父類汁讼,多重繼承就是淆攻,例如 A 類繼承 B 類,B 類繼承 C 類嘿架,所以按照關(guān)系就是 C 類是 B 類的父類瓶珊,B 類是 A 類的父類,這是 Java 繼承區(qū)別于 C++ 繼承的一個特性耸彪。
提高了類之間的耦合性(繼承的缺點伞芹,耦合度高就會造成代碼之間的聯(lián)系越緊密,代碼獨立性越差)蝉娜。
繼承關(guān)鍵字
繼承可以使用 extends 和 implements 這兩個關(guān)鍵字來實現(xiàn)繼承唱较,而且所有的類都是繼承于 java.lang.Object,當(dāng)一個類沒有繼承的兩個關(guān)鍵字召川,則默認(rèn)繼承object(這個類在?java.lang?包中绊汹,所以不需要?import)祖先類。
extends關(guān)鍵字
在 Java 中扮宠,類的繼承是單一繼承西乖,也就是說,一個子類只能擁有一個父類坛增,所以 extends 只能繼承一個類获雕。
extends 關(guān)鍵字
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{ }
implements關(guān)鍵字
使用 implements 關(guān)鍵字可以變相的使java具有多繼承的特性,使用范圍為類繼承接口的情況收捣,可以同時繼承多個接口(接口跟接口之間采用逗號分隔)届案。
implements 關(guān)鍵字
public interface A { public void eat(); public void sleep(); } public interface B { public void show(); } public class C implements A,B { }
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 :eatdog :eatanimal :eat
final關(guān)鍵字
final 關(guān)鍵字聲明類可以把類定義為不能繼承的尽纽,即最終類;或者用于修飾方法童漩,該方法不能被子類重寫:
聲明類:final class 類名 {//類體}
聲明方法:修飾符(public/private/default/protected) final 返回值類型 方法名(){//方法體}
注:實例變量也可以被定義為 final弄贿,被定義為 final 的變量不能被修改。被聲明為 final 類的方法自動地聲明為 final矫膨,但是實例變量并不是 final
構(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()SubClassSuperClass(int n)SubClass(int n):100------SubClass2 類繼承------SuperClass(int n)SubClass2SuperClass()SubClass2(int n):200