深入理解Java中的重寫和重載
重載(Overloading)和重寫(Overriding)是Java中兩個比較重要的概念。但是對於新手來說也比較容易混淆。本文通過兩個簡單的例子說明了他們之間的區別。
定義
重載
簡單說,就是函數或者方法有同樣的名稱,但是參數列表不相同的情形,這樣的同名不同參數的函數或者方法之間,互相稱之爲重載函數或者方法。
重寫
重寫指的是在Java的子類與父類中有兩個名稱、參數列表都相同的方法的情況。由於他們具有相同的方法簽名,所以子類中的新方法將覆蓋父類中原有的方法。
重載 VS 重寫
關於重載和重寫,你應該知道以下幾點:
1、重載是一個編譯期概念、重寫是一個運行期間概念。
2、重載遵循所謂“編譯期綁定”,即在編譯時根據參數變量的類型判斷應該調用哪個方法。
3、重寫遵循所謂“運行期綁定”,即在運行的時候,根據引用變量所指向的實際對象的類型來調用方法
4、因爲在編譯期已經確定調用哪個方法,所以重載並不是多態。而重寫是多態。重載只是一種語言特性,是一種語法規則,與多態無關,與面向對象也無關。(注:嚴格來說,重載是編譯時多態,即靜態多態。但是,Java中提到的多態,在不特別說明的情況下都指動態多態)
重寫的例子
下面是一個重寫的例子,看完代碼之後不妨猜測一下輸出結果:
class Dog{
public void bark(){
System.out.println("woof ");
}
}
class Hound extends Dog{
public void sniff(){
System.out.println("sniff ");
}
public void bark(){
System.out.println("bowl");
}
}
public class OverridingTest{
public static void main(String [] args){
Dog dog = new Hound();
dog.bark();
}
}
輸出結果:
bowl
上面的例子中,dog
對象被定義爲Dog
類型。在編譯期,編譯器會檢查Dog類中是否有可訪問的bark()
方法,只要其中包含bark()
方法,那麼就可以編譯通過。在運行期,Hound
對象被new
出來,並賦值給dog
變量,這時,JVM是明確的知道dog
變量指向的其實是Hound
對象的引用。所以,當dog
調用bark()
方法的時候,就會調用Hound
類中定義的bark()
方法。這就是所謂的動態多態性。
重寫的條件(規則)
1. 參數列表必須完全與被重寫方法的相同;
2. 返回類型必須完全與被重寫方法的返回類型相同;
3. 訪問級別的限制性一定不能比被重寫方法的強;(public>protected>default>private)
4. 訪問級別的限制性可以比被重寫方法的弱;
5. 重寫方法一定不能拋出新的檢查異常或比被重寫的方法聲明的檢查異常更廣泛的檢查異常;
6. 重寫的方法能夠拋出更少或更有限的異常(也就是說,被重寫的方法聲明瞭異常,但重寫的方法可以什麼也不聲明);
7. 不能重寫被標示爲final的方法;
8. 如果不能繼承一個方法,則不能重寫這個方法。
重載的例子
class Dog{
public void bark(){
System.out.println("woof ");
}
//overloading method
public void bark(int num){
for(int i=0; i<num; i++)
System.out.println("woof ");
}
}
上面的代碼中,定義了兩個bark方法,一個是沒有參數的bark方法,另外一個是包含一個int類型參數的bark方法。在編譯期,編譯期可以根據方法簽名(方法名和參數情況)情況確定哪個方法被調用。
重載的條件(規則)
1. 被重載的方法必須改變參數列表;
2. 被重載的方法可以改變返回類型;
3. 被重載的方法可以改變訪問修飾符;
4. 被重載的方法可以聲明新的或更廣的檢查異常;
5. 方法能夠在同一個類中或者在一個子類中被重載。
參考資料
Overriding vs. Overloading in Java