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