上篇回顧:多態是面向對象程序設計很是重要的特性,它讓程序擁有 更好的可讀性和可擴展性。ide
引用類型的變量具備兩種類型:編譯時類型和運行時類型。(也分別叫作聲明類型和實際類型)舉個簡單的例子:設計
//假設Student類是Person類的子類 Person p = new Student();
Person
即爲引用變量p的編譯時類型。Student
即爲引用變量p的運行時類型。將方法調用同方法主體關聯起來被稱爲綁定。code
在程序執行前進行綁定,叫作靜態綁定,也稱做前期綁定。在面向過程的語言中是默認的綁定方式。對象
在Java中,用private、static和final修飾的方法(static和final以後會作出總結)或構造器可以準確地讓編譯器調用哪一個方法,就是靜態綁定(static binding)。繼承
在運行時根據對象的運行時類型進行綁定,叫作動態綁定,也叫作後期綁定。固然在Java中,除了靜態綁定的那些方法,其餘方法的調用方式就是動態綁定啦。編譯器
public class DynamicBinding { //Object是全部類的超類,根據向上轉型,該方法能夠接受任何類型的對象 public static void test(Object x) { System.out.println(x.toString()); } public static void main(String[] args) { test(new PrimaryStudent());//Student test(new Student());//Student test(new Person());//Person test(new Object());//java.lang.Object@1b6d3586 } } class Person extends Object { @Override public String toString() { return "Person"; } public void run(){} public void count(int a){} } class Student extends Person { @Override public String toString() { return "Student"; } public void jump(){} } class PrimaryStudent extends Student { }
Object
。注意:引用變量只能調用編譯時類型所具備的方法。咱們還能夠發現,test(new PrimaryStudent());
的運行結果是Student
,,結果很明顯,由於PrimaryStudent
類中並無重寫父類的方法,若是採用動態綁定的方式調用方法,虛擬機會首先在本類中尋找適合的方法,若是沒有,會一直向父類尋找,直到找到爲止。虛擬機
那麼,每次調用時都要向上尋找,時間開銷必然會很大。爲此虛擬機預先爲每一個類都建立了方法表,其中列出了全部的方法簽名(返回值類型不算)和實際調用的方法,這樣子的話,在調用方法時直接查表就能夠了。(值得一提的是,若是用super限定調用父類方法,那麼將直接在實際類型的父類的表中查找)it
Person
類的方法表:Person: //下面省略Object方法簽名 //xxx()-> Object.xxx() //方法簽名->實際調用的方法 toString()->Person.toString() run()->Person.run() count(int)->Person(int)
Student
類的方法表:Student: //下面省略Object方法簽名 //xxx()-> Object.xxx() //方法簽名->實際調用的方法 toString()->Student.toString() jump()->Student.jump() run()->Person.run() count(int)->Person(int)
PrimaryStudent
類的方法表(PrimaryStudent
類爲空,直接繼承Student
類):PrimaryStudentt: //下面省略Object方法簽名 //xxx()-> Object.xxx() //方法簽名->實際調用的方法 toString()->Student.toString() jump()->Student.jump() run()->Person.run() count(int)->Person(int)
test(new PrimaryStudent());
語句時,虛擬機將會提取PrimaryStudent
的方法表。toString
簽名的類。這時虛擬機已經知道須要調用Student
類型的toString()
方法。動態綁定大大提高了程序的可擴展性,好比,我如今要新增一個
Teacher
類,能夠直接讓Teache
r類繼承於Person
類,再用Object
類的引用指向Teacher
對象,而不用作其餘的代碼調整,動態綁定自動搞定,就至關舒服。編譯
參考書籍:《Thinking in Java》、《Java核心技術卷I》