本文介紹了編譯器如何決定是重載還是重寫?的處理方法,對大家解決問題具有一定的參考價值,需要的朋友們下面隨著小編來一起學習吧!
問題描述
class A {
public void doSomething(float f) {
//print "in A"
}
}
class B extends A {
public void doSomething(int i) {
// print "in B"
}
public static void main(String[] args) {
A a = new B();
a.doSomething(10);
}
}
編譯器首先檢查doSomething()
是否存在于類A中,一旦確認,則在運行時檢查對象類型并執行相應的方法,即應該執行類B的doSomething()
,而執行類A
的doSomething()
。
如果兩個方法相同,則只執行B
doSomething()
類。
編譯器如何確定某個方法被重寫,從而由JVM處理?
推薦答案
簡單地說:
-
編譯器選擇要調用的方法的簽名
運行庫選擇將運行編譯器選擇的簽名的哪個實現
在步驟1中,編譯器查看調用該方法的對象的聲明類型,以及參數類型,這意味著:
編譯器只知道a
是A
(";聲明類型,又稱靜態類型);因此它只能搜索A
中的方法或A
繼承的方法。這意味著編譯器甚至不會在示例中的類B
中搜索doSomething
。
它查看參數的數據類型以在重載中解析一個方法。在您的示例中,這不是必需的,因為只有一個doSomething
方法(帶有float
參數的方法)
基于以上情況,編譯器只能得出將運行的doSomething(float)
的結論。您可以將其視為編譯器的選定方法簽名是一成不變的,運行庫無法更改這一點。
在步驟2中,運行庫知道將運行哪個簽名(doSomething(float)
),它唯一需要做的就是選擇將調用該簽名的哪個實現。為此,它查看了示例中的實際對象類型(創建的對象)new B()
。然后,它將在B
、中運行實現(如果被重寫),或者在樹中搜索該簽名的任何被重寫的實現,直到它到達A
中繼承的實現。由于B
不重寫doSomething(float)
,因此從A
繼承的實現A.doSomething(float)
運行。
這篇關于編譯器如何決定是重載還是重寫?的文章就介紹到這了,希望我們推薦的答案對大家有所幫助,