## Java編程那些事兒63—多態性
鄭州游戲學院陳躍峰
出自:[http://blog.csdn.net/mailbomb](http://blog.csdn.net/mailbomb)
#### 8.5.3 多態性
多態性是面向對象技術中最靈活的特性,主要是增強項目的可擴展性,提高代碼的可維護性。
多態性依賴繼承特性,可以把多態理解為繼承性的擴展或者深入。
在這里把多態性分為兩方面來進行介紹,對象類型的多態和對象方法的多態。
為了方便后續的講解,首先給出一個繼承結構的示例。
~~~
?????????????????? //文件名:SuperClass.java
?????????????????? public class SuperClass{
??????????????????????????? public void test(){
???????????????????????????????????? System.out.println(“SuperClass”);
??????????????????????????? }
?????????????????? }
?????????????????? // 文件名:SubbClass1.java
?????????????????? public class SubbClass1 extends SuperClass{
??????????????????????????? public void test(){
???????????????????????????????????? System.out.println(“SubbClass1”);
??????????????????????????? }
?????????????????? }
?????????????????? // 文件名:SubbClass2.java
?????????????????? public class SubbClass2 extends SuperClass{
??????????????????????????? public void test(){
???????????????????????????????????? System.out.println(“SubbClass2”);
??????????????????????????? }
?????????????????? }
~~~
在該示例代碼中,SubbClass1和SubbClass2是SuperClass的子類,并且在子類的內部都覆蓋父類中的test方法。由于這三個類中都書寫構造方法,則按照默認構造方法的約定,每個類中都會被自動添加一個默認的構造方法。
##### 8.5.3.1 對象類型的多態
對象類型的多態是指聲明對象的類型不是對象的真正類型,而對象的真正類型由創建對象時調用的構造方法進行決定。例外,按照繼承性的說明,子類的對象也是父類類型的對象,可以進行直接賦值。
例如如下代碼:
SuperClass sc = new SubbClass1();
這里聲明了一個SuperClass類型的對象sc,然后使用SuperClass的子類SubbClass1的構造方法進行創建,因為子類類型的對象也是父類類型的對象,所以創建出來的對象可以直接賦值給父類類型的對象sc。除了對象的賦值以外,另外一個更重要的知識是sc對象雖然使用SuperClass聲明的類型,但是內部存儲的卻是SubbClass1類型的對象。這個可以Java語言的中instanceof運算符進行判斷。
instanceof是一個運算符,其作用是判斷一個對象是否是某個類類型的對象,如果成立則表達式的值為true,否則為false。語法格式如下:
對象名 instanceof 類名
需要注意的是:這里的類名必須和聲明對象時的類之間存儲繼承關系,否則將出現語法錯誤。
測試類型的代碼如下:
~~~
?????????????????? /**
?* 測試對象類型的多態
?*/
public class TestObjectType {
???????? public static void main(String[] args) {
?????????????????? ?????????????????? SuperClass sc = new SubbClass1();
?????????????????? ?????????????????? boolean b = sc instanceof SuperClass;
?????????????????? ?????????????????? boolean b1 = sc instanceof SubbClass1;
?????????????????? ?????????????????? System.out.println(b);
?????????????????? ?????????????????? System.out.println(b1);
???????? ?????????????????? }
}
~~~
該測試程序的輸出結果是:
?????????????????? true
?????????????????? true
由程序運行結果可以看出,sc既是SuperClass類型的對象,也是SubbClass1類型的對象,而SubbClass1的類型被隱藏起來了,這就是對象的多態。其實sc對象不僅僅在類型上是SubbClass1類型的,其存儲的內容也是SubbClass1的內容,具體參看后面介紹的對象方法的多態。
對象類型的多態有很多的用途,極大的方便了對象的存儲和傳遞,使代碼很方便的進行擴展,對于已有代碼不產生影響。下面介紹兩個基本的使用。
1.對象的存儲
在存儲一系列不同子類的對象時,可以使用父類的結構來進行聲明,這樣可以方便數據的存儲,例如需要存儲多個SubbClass1和SubbClass2的對象時,則可以聲明一個SuperClass類型的數組進行存儲,示例代碼如下:
???????? SuperClass sc[] = new SuperClass[3];
???????? sc[0] = new SubbClass1();
???????? sc[1] = new SubbClass2();
???????? sc[2] = new SubbClass1();
則這里的數組sc,可以存儲各個類型子類的對象,而數組中每個元素的值都是存儲的對應子類的對象,而只是在名義上的類型(語法上的類型)是SuperClass類型的,這樣將方便程序的控制,當增加新的子類類型時,已有的代碼不需要進行改造就可以自動適應新的子類的結構。
例如新增了一個SuperClass的子類SubbClass3,則該數組的代碼可以修改成如下:
???????? SuperClass sc[] = new SuperClass[3];
???????? sc[0] = new SubbClass1();
???????? sc[1] = new SubbClass2();
???????? sc[2] = new SubbClass3();
其它的代碼都需要進行修改,就可以適應新的結構,這是多態性最主要的用途。
2.對象的傳遞
在方法的傳入參數傳遞,以及返回值處理方面都從對象類型的多態中受益。在向方法中傳入參數時,如果該方法需要處理各個子類的對象,則只需要書寫一個接受父類類型對象的方法即可。例如:
public void testObjectTypeMethod(SuperClass sc){}
則該在調用該方法時,可以傳入SuperClass的對象,也可以傳入其子類的對象,如果傳入的是子類的對象,則子類對象中的內容不會丟失。例如調用的示例代碼如下:
???????? SuperClass sc = new SuperClass();
???????? SubbClass1 sc1 = new SubbClass1();
???????? SubbClass2 sc2 = new SubbClass2();
???????? testObjectTypeMethod(sc);
testObjectTypeMethod(sc1);
testObjectTypeMethod(sc2);
這里說明的只是調用時的語法結構,這樣的特性將使我們只需要書寫一個方法,就可以處理所有子類的對象,簡化代碼的書寫,降低代碼的重復,從而降低維護的難度。
另外,方法的返回值也可以利用到該特性,例如如下方法:
public SuperClass testObjectTypeMethod2(){}
則在該方法的內部,既可以返回SuperClass類型的對象,也可以返回其子類的對象,也能簡化代碼的書寫,便于代碼的閱讀和維護。
關于對象類型的多態,就簡單的說明這么多,具體在項目中如何進行使用,還需要一定的技巧和方法。
##### 8.5.3.2 對象方法的多態
對象方法的多態基于方法的覆蓋,也就是該對象調用的方法具體是子類的方法還是父類的方法,由創建對象時使用的構造方法決定,而不是由聲明對象時聲明的類型決定。
示例代碼如下:
~~~
?????????????????? /**
???????????????? ?* 測試對象方法的多態
?*/
public class TestObjectMethod {
?????????????????? ???????? public static void main(String[] args) {
?????????????????? ?????????????????? SuperClass sc = new SuperClass();
?????????????????? ?????????????????? SubbClass1 sc1 = new SubbClass1();
?????????????????? ?????????????????? SubbClass2 sc2 = new SubbClass2();
???????????????????????????????????? SuperClass sc3 = new SubbClass1();
?????????????????? ?????????????????? testObjectTypeMethod(sc);
?????????????????? ?????????????????? testObjectTypeMethod(sc1);
?????????????????? ?????????????????? testObjectTypeMethod(sc2);
???????????????????????????????????? testObjectTypeMethod(sc3);
?????????????????? ???????? }
???????????????????????????
?????????????????? ???????? public static void testObjectTypeMethod(SuperClass sc){
?????????????????? ?????????????????? sc.test();? //調用被覆蓋的方法
?????????????????? ???????? }
}
???????? 該代碼的執行結果如下:
?????????????????? SuperClass
SubbClass1
SubbClass2
SubbClass1
~~~
則從代碼的執行結果看,雖然testObjectTypeMethod方法接收的是SuperClass類型的對象,但是傳入子類對象時,子類對象的內容沒有丟失,所以在調用test方法時,還是調用的對應對象中對應的test方法。
這樣就在功能上實現了對象的傳遞,從而保留了對象的內容,極大的方便了代碼的擴展性。
但是,由于Java在執行程序時,在程序運行的過程中,需要判斷對象調用的具體是父類的方法還是子類的方法,所以程序的執行速度會稍微有所降低。
- 前言
- (1)序言
- (2)程序設計是什么?
- (3)你適合學習程序設計嗎?
- (4)如何學好程序設計?
- (5)程序設計介紹小結
- (6)計算機軟件基本概念
- (7)進制的概念
- (8)計算機內部的數據表達
- (9)網絡編程基礎
- (10)Java語言簡介
- (11)JDK的獲得、安裝和配置
- (12)第一個HelloWorld程序
- (13)Eclipse基本使用
- (14)Eclipse基礎使用進階
- (15)如何學好Java語法
- (16)代碼框架、關鍵字和標識符
- (17)基本數據類型
- (18)變量和常量
- (19)數據類型轉換
- (20)空白、語句結束和注釋
- (21)算術運算符
- (22)比較運算符
- (23)邏輯運算符
- (24)賦值運算符
- (25)位運算符
- (26)移位運算符
- (27)其它運算符
- (28)運算符優先級
- (29)表達式
- (30)流程控制基礎
- (31)if語句語法(1)
- (32)if語句語法(2)
- (33)if語句語法(3)
- (34)switch語句語法
- (35)while語句語法
- (36)do-while語句語法
- (37)for語句語法
- (38)break和continue語句
- (39)流程控制綜合示例1
- (40)流程控制綜合示例2
- (41)流程控制綜合示例3
- (42)流程控制綜合練習
- (43)數組概述
- (44)數組基礎語法
- (45)數組使用示例1
- (46)數組使用示例2
- (47)數組使用示例3
- (48)多維數組基礎
- (49)多維數組使用示例1
- (50)多維數組使用示例2
- (51)多維數組練習
- (52)方法聲明
- (53)方法聲明示例
- (54)方法調用
- (55)方法重載和參數傳遞
- (56)方法練習
- (57)面向對象基礎
- (58)類(一)
- (59)類(二)
- (60)對象
- (61)面向對象設計方法和面向對象特性(一)
- (62)繼承(二)
- (63)多態性
- (64)訪問控制符、修飾符和其它關鍵字
- (65)static修飾符
- (66)final修飾符
- (67)this和super
- (68)抽象類和接口(一)
- (69)抽象類和接口(二)
- (70)抽象類和接口(三)
- (71)內部類簡介
- (72)包的概念
- (73)JDK文檔使用
- (74)java.lang包介紹1
- (75)String類使用
- (76)StringBuffer類和System類
- (77)包裝類
- (78)時間和日期處理
- (79)Random隨機處理
- (80)集合框架簡述
- (81)異常處理概述
- (82)異常處理語法1
- (83)異常處理語法2
- (84)IO簡介
- (85)IO類體系
- (86)文件操作之File類使用
- (87)文件操作之讀取文件
- (88)文件操作之寫文件
- (89)讀取控制臺輸入
- (90)裝飾流使用1
- (91)裝飾流使用2
- (92)IO使用注意問題
- (93)多線程基礎
- (94)多線程實現方式1
- (95)多線程實現方式2
- (96)多線程使用示例1
- (97)多線程使用示例2
- (98)多線程問題及處理1
- (99)多線程問題及處理2
- (100)多線程問題及處理3
- (101)網絡編程概述
- (102)網絡編程技術1
- (103)網絡編程技術2
- (104)網絡編程技術3
- (105)網絡編程技術4
- (106)網絡編程技術5
- (107)網絡協議概念
- (108)網絡編程示例1
- (109)網絡編程示例2
- (110)網絡編程小結