Java Override/Overload
重寫(xiě)(Override)
重寫(xiě)是子類(lèi)對(duì)父類(lèi)的允許訪問(wèn)的方法的實(shí)現(xiàn)過(guò)程進(jìn)行重新編寫(xiě)!返回值和形參都不能改變。即外殼不變,核心重寫(xiě)!
重寫(xiě)的好處在于子類(lèi)可以根據(jù)需要,定義特定于自己的行為。
也就是說(shuō)子類(lèi)能夠根據(jù)需要實(shí)現(xiàn)父類(lèi)的方法。
在面向?qū)ο笤瓌t里,重寫(xiě)意味著可以重寫(xiě)任何現(xiàn)有方法。實(shí)例如下:
class Animal{ public void move(){ System.out.println("動(dòng)物可以移動(dòng)"); } } class Dog extends Animal{ public void move(){ System.out.println("狗可以跑和走"); } } public class TestDog{ public static void main(String args[]){ Animal a = new Animal(); // Animal 對(duì)象 Animal b = new Dog(); // Dog 對(duì)象 a.move();// 執(zhí)行 Animal 類(lèi)的方法 b.move();//執(zhí)行 Dog 類(lèi)的方法 } }
以上實(shí)例編譯運(yùn)行結(jié)果如下:
動(dòng)物可以移動(dòng) 狗可以跑和走
在上面的例子中可以看到,盡管b屬于Animal類(lèi)型,但是它運(yùn)行的是Dog類(lèi)的move方法。
這是由于在編譯階段,只是檢查參數(shù)的引用類(lèi)型。
然而在運(yùn)行時(shí),Java虛擬機(jī)(JVM)指定對(duì)象的類(lèi)型并且運(yùn)行該對(duì)象的方法。
因此在上面的例子中,之所以能編譯成功,是因?yàn)锳nimal類(lèi)中存在move方法,然而運(yùn)行時(shí),運(yùn)行的是特定對(duì)象的方法。
思考以下例子:
class Animal{ public void move(){ System.out.println("動(dòng)物可以移動(dòng)"); } } class Dog extends Animal{ public void move(){ System.out.println("狗可以跑和走"); } public void bark(){ System.out.println("狗可以吠叫"); } } public class TestDog{ public static void main(String args[]){ Animal a = new Animal(); // Animal 對(duì)象 Animal b = new Dog(); // Dog 對(duì)象 a.move();// 執(zhí)行 Animal 類(lèi)的方法 b.move();//執(zhí)行 Dog 類(lèi)的方法 b.bark(); } }
以上實(shí)例編譯運(yùn)行結(jié)果如下:
TestDog.java:30: cannot find symbol symbol : method bark() location: class Animal b.bark(); ^
該程序?qū)伋鲆粋€(gè)編譯錯(cuò)誤,因?yàn)閎的引用類(lèi)型Animal沒(méi)有bark方法。
方法的重寫(xiě)規(guī)則
參數(shù)列表必須完全與被重寫(xiě)方法的相同;
返回類(lèi)型必須完全與被重寫(xiě)方法的返回類(lèi)型相同;
訪問(wèn)權(quán)限不能比父類(lèi)中被重寫(xiě)的方法的訪問(wèn)權(quán)限更低。例如:如果父類(lèi)的一個(gè)方法被聲明為public,那么在子類(lèi)中重寫(xiě)該方法就不能聲明為protected。
父類(lèi)的成員方法只能被它的子類(lèi)重寫(xiě)。
聲明為final的方法不能被重寫(xiě)。
聲明為static的方法不能被重寫(xiě),但是能夠被再次聲明。
子類(lèi)和父類(lèi)在同一個(gè)包中,那么子類(lèi)可以重寫(xiě)父類(lèi)所有方法,除了聲明為private和final的方法。
子類(lèi)和父類(lèi)不在同一個(gè)包中,那么子類(lèi)只能夠重寫(xiě)父類(lèi)的聲明為public和protected的非final方法。
重寫(xiě)的方法能夠拋出任何非強(qiáng)制異常,無(wú)論被重寫(xiě)的方法是否拋出異常。但是,重寫(xiě)的方法不能拋出新的強(qiáng)制性異常,或者比被重寫(xiě)方法聲明的更廣泛的強(qiáng)制性異常,反之則可以。
構(gòu)造方法不能被重寫(xiě)。
如果不能繼承一個(gè)方法,則不能重寫(xiě)這個(gè)方法。
Super關(guān)鍵字的使用
當(dāng)需要在子類(lèi)中調(diào)用父類(lèi)的被重寫(xiě)方法時(shí),要使用super關(guān)鍵字。
class Animal{ public void move(){ System.out.println("動(dòng)物可以移動(dòng)"); } } class Dog extends Animal{ public void move(){ super.move(); // 應(yīng)用super類(lèi)的方法 System.out.println("狗可以跑和走"); } } public class TestDog{ public static void main(String args[]){ Animal b = new Dog(); // Dog 對(duì)象 b.move(); //執(zhí)行 Dog類(lèi)的方法 } }
以上實(shí)例編譯運(yùn)行結(jié)果如下:
動(dòng)物可以移動(dòng) 狗可以跑和走
重載(Overload)
重載(overloading) 是在一個(gè)類(lèi)里面,方法名字相同,而參數(shù)不同。返回類(lèi)型可以相同也可以不同。
每個(gè)重載的方法(或者構(gòu)造函數(shù))都必須有一個(gè)獨(dú)一無(wú)二的參數(shù)類(lèi)型列表。
只能重載構(gòu)造函數(shù)
重載規(guī)則
被重載的方法必須改變參數(shù)列表;
被重載的方法可以改變返回類(lèi)型;
被重載的方法可以改變?cè)L問(wèn)修飾符;
被重載的方法可以聲明新的或更廣的檢查異常;
方法能夠在同一個(gè)類(lèi)中或者在一個(gè)子類(lèi)中被重載。
實(shí)例
public class Overloading { public int test(){ System.out.println("test1"); return 1; } public void test(int a){ System.out.println("test2"); } //以下兩個(gè)參數(shù)類(lèi)型順序不同 public String test(int a,String s){ System.out.println("test3"); return "returntest3"; } public String test(String s,int a){ System.out.println("test4"); return "returntest4"; } public static void main(String[] args){ Overloading o = new Overloading(); System.out.println(o.test()); o.test(1); System.out.println(o.test(1,"test3")); System.out.println(o.test("test4",1)); } }
重寫(xiě)與重載之間的區(qū)別
區(qū)別點(diǎn) | 重載方法 | 重寫(xiě)方法 |
---|---|---|
參數(shù)列表 | 必須修改 | 一定不能修改 |
返回類(lèi)型 | 可以修改 | 一定不能修改 |
異常 | 可以修改 | 可以減少或刪除,一定不能拋出新的或者更廣的異常 |
訪問(wèn) | 可以修改 | 一定不能做更嚴(yán)格的限制(可以降低限制) |