運行時多態(tài)性是面向?qū)ο蟪绦蛟O(shè)計代碼重用的一個最強大機制 動態(tài)性的概念也可以被說成 一個接口 多個方法 Java實現(xiàn)運行時多態(tài)性的基礎(chǔ)是動態(tài)方法調(diào)度 它是一種在運行時而不是在編譯期調(diào)用重載方法的機制 下面就繼承和接口實現(xiàn)兩方面談?wù)刯ava運行時多態(tài)性的實現(xiàn)
成都創(chuàng)新互聯(lián)專注于企業(yè)營銷型網(wǎng)站、網(wǎng)站重做改版、綏棱網(wǎng)站定制設(shè)計、自適應(yīng)品牌網(wǎng)站建設(shè)、H5建站、商城網(wǎng)站開發(fā)、集團公司官網(wǎng)建設(shè)、成都外貿(mào)網(wǎng)站建設(shè)公司、高端網(wǎng)站制作、響應(yīng)式網(wǎng)頁設(shè)計等建站業(yè)務(wù),價格優(yōu)惠性價比高,為綏棱等各大城市提供網(wǎng)站開發(fā)制作服務(wù)。
一 通過繼承中超類對象引用變量引用子類對象來實現(xiàn)
舉例說明
//定義超類superA
class superA
{
int i =
void fun()
{
System out println( This is superA )
}
}
//定義superA的子類subB
class subB extends superA
{
int m =
void fun()
{
System out println( This is subB )
}
}
//定義superA的子類subC
class subC extends superA
{
int n =
void fun()
{
System out println( This is subC )
}
}
class Test
{
public static void main(String[] args)
{
superA a
subB b = new subB()
subC c = new subC()
a=b
a fun() ???????? ( )
a=c
a fun() ???????? ( )
}
}
運行結(jié)果為
This is subB
This is subC
上述代碼中subB和subC是超類superA的子類 我們在類Test中聲明了 個引用變量a b c 通過將子類對象引用賦值給超類對象引用變量來實現(xiàn)動態(tài)方法調(diào)用 也許有人會問 為什么( )和( )不輸出 This is superA java 的這種機制遵循一個原則 當(dāng)超類對象引用變量引用子類對象時 被引用對象的類型而不是引用變量的類型決定了調(diào)用誰的成員方法 但是這個被調(diào)用的方法必須是在超類中定義過的 也就是說被子類覆蓋的方法
所以 不要被上例中( )和( )所迷惑 雖然寫成a fun() 但是由于( )中的a被b賦值 指向了子類subB的一個實例 因而( )所調(diào)用的fun()實際上是子類subB的成員方法fun() 它覆蓋了超類superA的成員方法fun() 同樣( )調(diào)用的是子類subC的成員方法fun()
另外 如果子類繼承的超類是一個抽象類 雖然抽象類不能通過new操作符實例化 但是可以創(chuàng)建抽象類的對象引用指向子類對象 以實現(xiàn)運行時多態(tài)性 具體的實現(xiàn)方法同上例
不過 抽象類的子類必須覆蓋實現(xiàn)超類中的所有的抽象方法 否則子類必須被abstract修飾符修飾 當(dāng)然也就不能被實例化了
二 通過接口類型變量引用實現(xiàn)接口的類的對象來實現(xiàn)
接口的靈活性就在于 規(guī)定一個類必須做什么 而不管你如何做 我們可以定義一個接口類型的引用變量來引用實現(xiàn)接口的類的實例 當(dāng)這個引用調(diào)用方法時 它會根據(jù)實際引用的類的實例來判斷具體調(diào)用哪個方法 這和上述的超類對象引用訪問子類對象的機制相似
舉例說明
//定義接口InterA
interface InterA
{
void fun()
}
//實現(xiàn)接口InterA的類B
class B implements InterA
{
public void fun()
{
System out println( This is B )
}
}
//實現(xiàn)接口InterA的類C
class C implements InterA
{
public void fun()
{
System out println( This is C )
}
}
class Test
{
public static void main(String[] args)
{
InterA a
a= new B()
a fun()
a = new C()
a fun()
}
}
輸出結(jié)果為
This is B
This is C
上例中類B和類C是實現(xiàn)接口InterA的兩個類 分別實現(xiàn)了接口的方法fun() 通過將類B和類C的實例賦給接口引用a而實現(xiàn)了方法在運行時的動態(tài)綁定 充分利用了 一個接口 多個方法 展示了Java的動態(tài)多態(tài)性
需要注意的一點是 Java在利用接口變量調(diào)用其實現(xiàn)類的對象的方法時 該方法必須已經(jīng)在接口中被聲明 而且在接口的實現(xiàn)類中該實現(xiàn)方法的類型和參數(shù)必須與接口中所定義的精確匹配
lishixinzhi/Article/program/Java/hx/201311/26085
因為new SuberClass();這段代碼,是實例化SuberClass類,就是調(diào)用SuberClass類的無參構(gòu)造函數(shù),但是SuberClass類繼承了BaseClass類,所以默認(rèn)先調(diào)用父類的構(gòu)造函數(shù)(
public BaseClass(){
System.out.println("Now in BaseClass()");
init();
}
)
有一個比較經(jīng)典的多態(tài)實例:
有一個Animal類,它有Cat,和Dog兩個子類,在Animal中有個say方法,當(dāng)Cat調(diào)用這個方法的時候輸出的是“小貓喵喵喵”,當(dāng)Dog調(diào)用這個方法時,輸出的是“小狗汪汪汪”,這就是Java多態(tài)的實現(xiàn)。
1、定義一種動物,該類型的動物有叫的屬性。
2、分別定義貓,狗,鳥,都繼承于該動物,都有叫的屬性。
3、分別表示出各個具體小動物的叫聲,例如貓的叫聲:喵、狗的叫聲:汪、鳥的叫聲:咻,點是叫聲,實現(xiàn)各個具體小動物用的叫聲的方法是用的同一個函數(shù)名稱,就是動物的叫聲函數(shù)。
多態(tài):
這個案例網(wǎng)上是沒有的,屬于無忌獨創(chuàng),當(dāng)時在幫孩子輔導(dǎo)作業(yè),小學(xué)科學(xué),里面有一點內(nèi)容是關(guān)于人的牙齒,說牙齒分為:門齒、犬齒、臼齒。
問閨女,為什么這么分呢?閨女說牙齒雖然都是用來咬食物,但是食物種類很多,咬碎需要的工具也不一樣,門齒用來切割食物,如:蘋果、梨;犬齒用來撕碎食物。
如肉類;臼齒用來磨碎食物,如犬齒撕碎的肉類,就需要再用臼齒來磨碎,然后送到食道,胃,小腸,大腸進行消化。我恍然大悟,這不就是Java面向?qū)ο罄锏亩鄳B(tài)嗎?多完美啊。
這也很好說明了為什么會有多態(tài)出現(xiàn),因為生活中就存在太多這種例子,所以需要多態(tài)來匹配解決。
第一步:創(chuàng)建一個父類,如Animal(動物) 類,這個方法中只有方法的定義,沒有實現(xiàn)。第二步:創(chuàng)建一個子類,如cat(貓)類,這個方法中實現(xiàn)父類中的所有方法,并進行重寫。
第三步:創(chuàng)建一個子類,如dog(狗)類,這個方法中實現(xiàn)父類中的所有方法,并進行重寫。
備注:多態(tài)實際上是一個父類有特定的屬性,如動物有名稱、年齡、所屬科目,之后由子類去實現(xiàn)具體的,如貓,名稱是貓,年齡是5歲,所屬科目為貓科,然后狗同樣有這些屬性,只不過是值不同,這個就是一個多態(tài)的形式。子類可以有無限多,在步驟三后面添加即可。