更新時(shí)間:2021-05-18 來源:黑馬程序員 瀏覽量:
當(dāng)子類重寫父類的方法后,子類對(duì)象將無法直接訪問父類被重寫的方法。為了解決這個(gè)問題,在Java中專門提供了一個(gè)super關(guān)鍵字來訪問父類的成員,例如訪問父類的成員變量、成員方法和構(gòu)造方法。下面分兩種情況來學(xué)習(xí)一下super關(guān)鍵字的具體用法。
(1)使用super關(guān)鍵字調(diào)用父類的成員變量和成員方法,具體格式如下:
super.成員變量 super.成員方法([參數(shù)1,參數(shù)2...])接下來通過一個(gè)案例來學(xué)習(xí)如何使用super關(guān)鍵字調(diào)用父類的成員變量和成員方法,如文件1所示。
文件1 Example03.java
// 定義Animal類 class Animal { String name = "動(dòng)物"; // 定義動(dòng)物叫的方法 void shout() { System.out.println("動(dòng)物發(fā)出叫聲"); } } // 定義Dog類繼承動(dòng)物類 class Dog extends Animal { String name = "犬類"; // 重寫父類的shout()方法 void shout() { super.shout(); // 訪問父類的成員方法 } // 定義打印name的方法 void printName() { System.out.println("name=" + super.name);// 訪問父類的成員變量 } } // 定義測(cè)試類 public class Example03{ public static void main(String[] args) { Dog dog = new Dog(); // 創(chuàng)建一個(gè)dog對(duì)象 dog.shout(); // 調(diào)用dog對(duì)象重寫的shout()方法 dog.printName(); // 調(diào)用dog對(duì)象的的printName()方法 } }
運(yùn)行結(jié)果如圖1所示。
圖1 運(yùn)行結(jié)果
文件1中,定義了一個(gè)Dog類繼承Animal類,重寫了Animal類的shout()方法并重新定義了子類的name屬性。在子類Dog的shout()方法中使用“super.shout()”調(diào)用了父類被重寫的方法,在printName()方法中使用“super.name”訪問父類的成員變量。從運(yùn)行結(jié)果可以看出,子類通過super關(guān)鍵字成功地訪問了父類成員變量和成員方法。
(2)使用super關(guān)鍵字調(diào)用父類的構(gòu)造方法,具體格式如下:
super([參數(shù)1,參數(shù)2...])
接下來就通過一個(gè)案例來學(xué)習(xí),如何使用super關(guān)鍵字來調(diào)用父類的構(gòu)造方法,如文件2所示。
文件2 Example04.java
// 定義Animal類 class Animal { // 定義Animal類有參的構(gòu)造方法 public Animal(String name) { System.out.println("我是一只" + name); } } // 定義Dog類繼承Animal類 class Dog extends Animal { public Dog() { super("沙皮狗"); // 調(diào)用父類有參的構(gòu)造方法 } } // 定義測(cè)試類 public class Example04 { public static void main(String[] args) { Dog dog = new Dog(); // 創(chuàng)建Dog類的實(shí)例對(duì)象 } }
運(yùn)行結(jié)果如圖2所示。
圖2 運(yùn)行結(jié)果
根據(jù)前面所學(xué)的知識(shí),文件1中在創(chuàng)建Dog類對(duì)象時(shí)一定會(huì)調(diào)用Dog類的構(gòu)造方法,從運(yùn)行結(jié)果可以看出,Dog類的構(gòu)造方法被調(diào)用時(shí),執(zhí)行了內(nèi)部的super("沙皮狗")方法,從而調(diào)用了父類的有參構(gòu)造方法。需要注意的是,通過super調(diào)用父類構(gòu)造方法的代碼必須位于子類構(gòu)造方法的第一行,并且只能出現(xiàn)一次,否則程序在編譯期間就會(huì)報(bào)錯(cuò)。
將文件1第11行代碼進(jìn)行注釋,程序就會(huì)出現(xiàn)編譯錯(cuò)誤,如圖3所示。
圖3 運(yùn)行結(jié)果
從圖3可以看出,程序編譯出現(xiàn)錯(cuò)誤,顯示“Implicit super constructor Animal() is undefined. Must explicitly invoke another constructor(未定義隱式無參構(gòu)造方法,必須顯示的調(diào)用另一個(gè)構(gòu)造方法)”的錯(cuò)誤。出錯(cuò)的原因是,在子類的構(gòu)造方法中一定會(huì)調(diào)用父類的某個(gè)構(gòu)造方法。這時(shí)可以在子類的構(gòu)造方法中通過super關(guān)鍵字指定調(diào)用父類的哪個(gè)構(gòu)造方法,如果沒有指定,在實(shí)例化子類對(duì)象時(shí),會(huì)默認(rèn)調(diào)用父類無參的構(gòu)造方法,而在文件2中,父類Animal中只定義了有參構(gòu)造方法,未定義無參構(gòu)造方法,所以在子類默認(rèn)調(diào)用父類無參構(gòu)造方法時(shí)就會(huì)出錯(cuò)。
為了解決上述程序的編譯錯(cuò)誤,可以在子類中顯示地調(diào)用父類中已有的構(gòu)造方法,或者在父類中定義無參的構(gòu)造方法?,F(xiàn)將文件2中的Animal類進(jìn)行修改,在父類中添加無參構(gòu)造方法來解決上述編譯錯(cuò)誤,如文件3所示。
文件3 Example05.java
// 定義Animal類 class Animal { // 定義Animal無參的構(gòu)造方法 public Animal() { System.out.println("我是一只動(dòng)物"); } // 定義Animal有參的構(gòu)造方法 public Animal(String name) { System.out.println("我是一只" + name); } } // 定義Dog類,繼承自Animal類 class Dog extends Animal { // 定義Dog類無參的構(gòu)造方法 public Dog() { } } // 定義測(cè)試類 public class Example05 { public static void main(String[] args) { Dog dog = new Dog(); // 創(chuàng)建Dog類的實(shí)例對(duì)象 } }
運(yùn)行結(jié)果如圖4所示。
圖4 運(yùn)行結(jié)果
從圖4可以看出,子類在實(shí)例化時(shí)默認(rèn)調(diào)用了父類無參的構(gòu)造方法。通過這個(gè)案例還可以得出一個(gè)結(jié)論:在定義一個(gè)類時(shí),如果沒有特殊需求,當(dāng)定義了有參構(gòu)造方法后,盡量在類中再顯示地定義一個(gè)無參構(gòu)造方法,這樣可以避免該類被繼承時(shí)出現(xiàn)錯(cuò)誤。
猜你喜歡:
詳解this關(guān)鍵字的用法【Java技術(shù)文章】