一個Java程式的執行要經過編譯和執行(解釋)這兩個步驟,同時Java又是物件導向的程式語言。當子類別和父類別存在同一個方法,子類別重寫了父類別的方法,程式在執行時呼叫方法是呼叫父類別的方法還是子類別的重寫方法呢,這應該是我們在初學Java時遇到的問題。這裡首先我們將確定這種呼叫何種方法實作或變數的操作叫做綁定。
在Java中存在兩種綁定方式,一種為靜態綁定,又稱作早期綁定。另一種是動態綁定,也稱為後期綁定。
程式綁定的概念:
綁定指的是一個方法的呼叫與方法所在的類別(方法主體)關聯起來。對java來說,綁定分為靜態綁定和動態綁定;或者稱為前期綁定和後期綁定
靜態綁定(早期綁定編譯器綁定):
在程式執行前方法已經被綁定,此時由編譯器或其它連接程式實作。例如:C 。針對java可以理解為程式編譯期的綁定;特別說明一點,java當中的方法只有final,static,private和構造方法是前期綁定
動態綁定(遲綁定運行期綁定):
後期綁定:在運行時根據特定物件的類型進行綁定。
若一種語言實現了後期綁定,同時必須提供一些機制在運行期間判斷對象的類型,並分別調用適當的方法。也就是說編譯器此時依然不知道物件的類型,但方法呼叫機制能自己去調查,找到正確的方法主體。不同的語言對後期綁定的實作方法是有所區別的。可以這樣認為:它們都要在物件中安插某些特殊類型的信息。
動態綁定的過程:
虛擬機提取物件的實際類型的方法表
虛擬機搜尋方法簽章
呼叫方法
虛擬機搜尋方法簽章呼叫方法關於綁定相關的總結:發現java屬於後期綁定。在java中,幾乎所有的方法都是後期綁定,在運行時動態綁定方法屬於子類別還是基類。但也有特殊,針對static方法和final方法由於不能被繼承,因此在編譯時就可以確定他們的值,他們是屬於前期綁定。特別說明的一點,private宣告的方法和成員變數不能被子類別繼承,所有的private方法都被隱式的指定為final的(由此我們知道:將方法宣告為final類型的 一是為了防止方法被覆寫,二是為了有效的關閉java中的動態綁定)。 java中的後期綁定是由JVM來實現的,我們不用去顯式的聲明它,而C++則不同,必須明確的聲明某個方法具備後期綁定。 java當中的向上轉型或說多態是藉助於動態綁定實現的,所以理解動態綁定,也就搞定向上轉型和多態。 對於java當中的方法而言,除了final,static,private和構造方法是前期綁定外,其他的方法全部為動態綁定。而動態綁定的典型發生在父類別和子類別的轉換聲明之下:例如:Parent p = new Children();具體過程如下:1、編譯器檢查物件的聲明類型和方法名。假設我們呼叫x.f(args)方法,並且x已經被宣告為C類別的對象,那麼編譯器會列舉出C類別中所有的名稱為f的方法和從C類別的超類別繼承過來的f方法 2、接下來編譯器檢查方法呼叫中提供的參數類型。如果在所有名稱為f 的方法中有一個參數類型和呼叫提供的參數類型最為匹配,那麼就調用這個方法,這個過程叫做“重載解析”3、當程式運行並且使用動態綁定呼叫方法時,虛擬機器必須呼叫同x所指向的物件的實際類型相符的方法版本。假設實際型別為D(C的子類),如果D類定義了f(String)那麼該方法被調用,否則就在D的超類中搜尋方法f(String),依次類推問題思考:如何提供給方法使用者一個方法,來完成一個任務。如果用戶有特殊要求又要可以自訂自己的方法? 涉及知識:子父類,接口,向上轉型,動態綁定具體代碼:
package com.chengxuyuanzhilu; public interface MyInterfaces { void doting(); } package com.chengxuyuanzhilu; public class Drink implements MyInterfaces { @Override public void doting() { System.out.println("我在喝水"); } } package com.chengxuyuanzhilu; public class Eat implements MyInterfaces { @Override public void doting() { System.out.println("我在吃东西"); } } package com.chengxuyuanzhilu; public class Run implements MyInterfaces { @Override public void doting() { System.out.println("我在奔跑"); } } package com.chengxuyuanzhilu; public class TestDynamicBind { public static void main(String[] args) { MyInterfaces my = null; my = new Eat(); bind(my); my = new Drink(); bind(my); my = new Run(); bind(my); } static void bind(MyInterfaces my){ my.doting(); } }