Java 覆蓋/重載
重寫(Override)
重寫是子類別對父類別的允許存取的方法的實作過程進行重新編寫!傳回值和形參都不能改變。即外殼不變,核心重寫!
重寫的好處是子類別可以根據需要,定義特定於自己的行為。
也就是說子類別能夠根據需要實作父類別的方法。
在物件導向原則裡,重寫意味著可以重寫任何現有方法。實例如下:
class Animal{ public void move(){ System.out.println("动物可以移动"); } } class Dog extends Animal{ public void move(){ System.out.println("狗可以跑和走"); } } public class TestDog{ public static void main(String args[]){ Animal a = new Animal(); // Animal 对象 Animal b = new Dog(); // Dog 对象 a.move();// 执行 Animal 类的方法 b.move();//执行 Dog 类的方法 } }
以上實例編譯運行結果如下:
动物可以移动 狗可以跑和走
在上面的例子中可以看到,儘管b屬於Animal類型,但是它運行的是Dog類的move方法。
這是由於在編譯階段,只是檢查參數的參考型別。
然而在執行時,Java虛擬機器(JVM)指定物件的類型並且執行該物件的方法。
因此在上面的例子中,之所以能編譯成功,是因為Animal類別中存在move方法,然而運行時,運行的是特定物件的方法。
思考以下例子:
class Animal{ public void move(){ System.out.println("动物可以移动"); } } 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 对象 Animal b = new Dog(); // Dog 对象 a.move();// 执行 Animal 类的方法 b.move();//执行 Dog 类的方法 b.bark(); } }
以上實例編譯運行結果如下:
TestDog.java:30: cannot find symbol symbol : method bark() location: class Animal b.bark(); ^
該程式將拋出一個編譯錯誤,因為b的引用類型Animal沒有bark方法。
方法的重寫規則
參數清單必須與重寫方法完全相同的;
回傳型別必須完全與重寫方法的回傳型別相同;
存取權限不能比父類別中被重寫的方法的存取權限更低。例如:如果父類別的一個方法被宣告為public,那麼在子類別中重寫該方法就不能宣告為protected。
父類別的成員方法只能被它的子類別重寫。
宣告為final的方法不能被重寫。
聲明為static的方法不能重寫,但能夠再次被聲明。
子類別和父類別在同一個套件中,那麼子類別可以重寫父類別所有方法,除了宣告為private和final的方法。
子類別和父類別不在同一個套件中,那麼子類別只能夠重寫父類別的宣告為public和protected的非final方法。
重寫的方法能夠拋出任何非強制異常,無論被重寫的方法是否拋出異常。但是,重寫的方法不能拋出新的強制性異常,或者比被重寫方法聲明的更廣泛的強制性異常,反之則可以。
構造方法不能被重寫。
如果不能繼承一個方法,則不能重寫這個方法。
Super關鍵字的使用
當需要在子類別中呼叫父類別的被重寫方法時,要使用super關鍵字。
class Animal{ public void move(){ System.out.println("动物可以移动"); } } class Dog extends Animal{ public void move(){ super.move(); // 应用super类的方法 System.out.println("狗可以跑和走"); } } public class TestDog{ public static void main(String args[]){ Animal b = new Dog(); // Dog 对象 b.move(); //执行 Dog类的方法 } }
以上實例編譯運行結果如下:
动物可以移动 狗可以跑和走
重載(Overload)
重載(overloading) 是在一個類別裡面,方法名字相同,而參數不同。傳回類型可以相同也可以不同。
每個重載的方法(或建構子)都必須有一個獨一無二的參數型別清單。
只能重載建構子
重載規則
#被重載的方法必須改變參數清單;
- #被重載的方法可以改變傳回型別;
被重載的方法可以宣告新的或更廣的檢查異常;
實例 | public class Overloading { public int test(){ System.out.println("test1"); return 1; } public void test(int a){ System.out.println("test2"); } //以下两个参数类型顺序不同 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)); } } | |
---|---|---|
##區別點 | 重載方法 | |
參數清單 | 必須修改 | |
回傳型別 | 可以修改 | |
異常 | 可以修改 |