国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服務器之家 - 編程語言 - JAVA教程 - 詳解Java多態對象的類型轉換與動態綁定

詳解Java多態對象的類型轉換與動態綁定

2020-01-06 14:39goldensun JAVA教程

這篇文章主要介紹了詳解Java多態對象的類型轉換與動態綁定,是Java入門學習中的基礎知識,需要的朋友可以參考下

Java多態對象的類型轉換
這里所說的對象類型轉換,是指存在繼承關系的對象,不是任意類型的對象。當對不存在繼承關系的對象進行強制類型轉換時,java 運行時將拋出 java.lang.ClassCastException 異常。

在繼承鏈中,我們將子類向父類轉換稱為“向上轉型”,將父類向子類轉換稱為“向下轉型”。

很多時候,我們會將變量定義為父類的類型,卻引用子類的對象,這個過程就是向上轉型。程序運行時通過動態綁定來實現對子類方法的調用,也就是多態性。

然而有些時候為了完成某些父類沒有的功能,我們需要將向上轉型后的子類對象再轉成子類,調用子類的方法,這就是向下轉型。

注意:不能直接將父類的對象強制轉換為子類類型,只能將向上轉型后的子類對象再次轉換為子類類型。也就是說,子類對象必須向上轉型后,才能再向下轉型。請看下面的代碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
public class Demo {
  public static void main(String args[]) {
    SuperClass superObj = new SuperClass();
    SonClass sonObj = new SonClass();
    // 下面的代碼運行時會拋出異常,不能將父類對象直接轉換為子類類型
    // SonClass sonObj2 = (SonClass)superObj;
    // 先向上轉型,再向下轉型
    superObj = sonObj;
    SonClass sonObj1 = (SonClass)superObj;
  }
}
class SuperClass{ }
class SonClass extends SuperClass{ }

將第7行的注釋去掉,運行時會拋出異常,但是編譯可以通過。

因為向下轉型存在風險,所以在接收到父類的一個引用時,請務必使用 instanceof 運算符來判斷該對象是否是你所要的子類,請看下面的代碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public class Demo {
  public static void main(String args[]) {
    SuperClass superObj = new SuperClass();
    SonClass sonObj = new SonClass();
    // superObj 不是 SonClass 類的實例
    if(superObj instanceof SonClass){
      SonClass sonObj1 = (SonClass)superObj;
    }else{
      System.out.println("①不能轉換");
    }
    superObj = sonObj;
    // superObj 是 SonClass 類的實例
    if(superObj instanceof SonClass){
      SonClass sonObj2 = (SonClass)superObj;
    }else{
      System.out.println("②不能轉換");
    }
  }
}
class SuperClass{ }
class SonClass extends SuperClass{ }

運行結果:

?
1
①不能轉換

總結:對象的類型轉換在程序運行時檢查,向上轉型會自動進行,向下轉型的對象必須是當前引用類型的子類。

Java多態和動態綁定
在Java中,父類的變量可以引用父類的實例,也可以引用子類的實例。

請讀者先看一段代碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
public class Demo {
  public static void main(String[] args){
    Animal obj = new Animal();
    obj.cry();
    obj = new Cat();
    obj.cry();
    obj = new Dog();
    obj.cry();
  }
}
class Animal{
  // 動物的叫聲
  public void cry(){
    System.out.println("不知道怎么叫");
  }
  
}
class Cat extends Animal{
  // 貓的叫聲
  public void cry(){
    System.out.println("喵喵~");
  }
}
class Dog extends Animal{
  // 狗的叫聲
  public void cry(){
    System.out.println("汪汪~");
  }
}

運行結果:

?
1
2
3
不知道怎么叫
喵喵~
汪汪~

上面的代碼,定義了三個類,分別是 Animal、Cat 和 Dog,Cat 和 Dog 類都繼承自 Animal 類。obj 變量的類型為 Animal,它既可以指向 Animal 類的實例,也可以指向 Cat 和 Dog 類的實例,這是正確的。也就是說,父類的變量可以引用父類的實例,也可以引用子類的實例。注意反過來是錯誤的,因為所有的貓都是動物,但不是所有的動物都是貓。

可以看出,obj 既可以是人類,也可以是貓、狗,它有不同的表現形式,這就被稱為多態。多態是指一個事物有不同的表現形式或形態。

再比如“人類”,也有很多不同的表達或實現,TA 可以是司機、教師、醫生等,你憎恨自己的時候會說“下輩子重新做人”,那么你下輩子成為司機、教師、醫生都可以,我們就說“人類”具備了多態性。

多態存在的三個必要條件:要有繼承、要有重寫、父類變量引用子類對象。

當使用多態方式調用方法時:
首先檢查父類中是否有該方法,如果沒有,則編譯錯誤;如果有,則檢查子類是否覆蓋了該方法。
如果子類覆蓋了該方法,就調用子類的方法,否則調用父類方法。

從上面的例子可以看出,多態的一個好處是:當子類比較多時,也不需要定義多個變量,可以只定義一個父類類型的變量來引用不同子類的實例。請再看下面的一個例子:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
public class Demo {
  public static void main(String[] args){
    // 借助多態,主人可以給很多動物喂食
    Master ma = new Master();
    ma.feed(new Animal(), new Food());
    ma.feed(new Cat(), new Fish());
    ma.feed(new Dog(), new Bone());
  }
}
// Animal類及其子類
class Animal{
  public void eat(Food f){
    System.out.println("我是一個小動物,正在吃" + f.getFood());
  }
}
class Cat extends Animal{
  public void eat(Food f){
    System.out.println("我是一只小貓咪,正在吃" + f.getFood());
  }
}
class Dog extends Animal{
  public void eat(Food f){
    System.out.println("我是一只狗狗,正在吃" + f.getFood());
  }
}
// Food及其子類
class Food{
  public String getFood(){
    return "事物";
  }
}
class Fish extends Food{
  public String getFood(){
    return "魚";
  }
}
class Bone extends Food{
  public String getFood(){
    return "骨頭";
  }
}
// Master類
class Master{
  public void feed(Animal an, Food f){
    an.eat(f);
  }
}

運行結果:

?
1
2
3
我是一個小動物,正在吃事物
我是一只小貓咪,正在吃魚
我是一只狗狗,正在吃骨頭

Master 類的 feed 方法有兩個參數,分別是 Animal 類型和 Food 類型,因為是父類,所以可以將子類的實例傳遞給它,這樣 Master 類就不需要多個方法來給不同的動物喂食。
動態綁定

為了理解多態的本質,下面講一下Java調用方法的詳細流程。

1) 編譯器查看對象的聲明類型和方法名。

假設調用 obj.func(param),obj 為 Cat 類的對象。需要注意的是,有可能存在多個名字為func但參數簽名不一樣的方法。例如,可能存在方法 func(int) 和 func(String)。編譯器將會一一列舉所有 Cat 類中名為func的方法和其父類 Animal 中訪問屬性為 public 且名為func的方法。

這樣,編譯器就獲得了所有可能被調用的候選方法列表。

2) 接下來,編澤器將檢查調用方法時提供的參數簽名。

如果在所有名為func的方法中存在一個與提供的參數簽名完全匹配的方法,那么就選擇這個方法。這個過程被稱為重載解析(overloading resolution)。例如,如果調用 func("hello"),編譯器會選擇 func(String),而不是 func(int)。由于自動類型轉換的存在,例如 int 可以轉換為 double,如果沒有找到與調用方法參數簽名相同的方法,就進行類型轉換后再繼續查找,如果最終沒有匹配的類型或者有多個方法與之匹配,那么編譯錯誤。

這樣,編譯器就獲得了需要調用的方法名字和參數簽名。

3) 如果方法的修飾符是private、static、final(static和final將在后續講解),或者是構造方法,那么編譯器將可以準確地知道應該調用哪個方法,我們將這種調用方式 稱為靜態綁定(static binding)。

與此對應的是,調用的方法依賴于對象的實際類型, 并在運行時實現動態綁。例如調用 func("hello"),編澤器將采用動態綁定的方式生成一條調用 func(String) 的指令。

4)當程序運行,并且釆用動態綁定調用方法時,JVM一定會調用與 obj 所引用對象的實際類型最合適的那個類的方法。我們已經假設 obj 的實際類型是 Cat,它是 Animal 的子類,如果 Cat 中定義了 func(String),就調用它,否則將在 Animal 類及其父類中尋找。

每次調用方法都要進行搜索,時間開銷相當大,因此,JVM預先為每個類創建了一個方法表(method lable),其中列出了所有方法的名稱、參數簽名和所屬的類。這樣一來,在真正調用方法的時候,虛擬機僅查找這個表就行了。在上面的例子中,JVM 搜索 Cat 類的方法表,以便尋找與調用 func("hello") 相匹配的方法。這個方法既有可能是 Cat.func(String),也有可能是 Animal.func(String)。注意,如果調用super.func("hello"),編譯器將對父類的方法表迸行搜索。

假設 Animal 類包含cry()、getName()、getAge() 三個方法,那么它的方法表如下:
cry() -> Animal.cry()
getName() -> Animal.getName()
getAge() -> Animal.getAge()

實際上,Animal 也有默認的父類 Object(后續會講解),會繼承 Object 的方法,所以上面列舉的方法并不完整。

假設 Cat 類覆蓋了 Animal 類中的 cry() 方法,并且新增了一個方法 climbTree(),那么它的參數列表為:
cry() -> Cat.cry()
getName() -> Animal.getName()
getAge() -> Animal.getAge()
climbTree() -> Cat.climbTree()

在運行的時候,調用 obj.cry() 方法的過程如下:
JVM 首先訪問 obj 的實際類型的方法表,可能是 Animal 類的方法表,也可能是 Cat 類及其子類的方法表。
JVM 在方法表中搜索與 cry() 匹配的方法,找到后,就知道它屬于哪個類了。
JVM 調用該方法。

延伸 · 閱讀

精彩推薦
  • JAVA教程Java枚舉類enum介紹

    Java枚舉類enum介紹

    這篇文章主要介紹了Java枚舉類enum介紹,和其它普通類一樣,enum同樣可以有成員變量、方法、構造器,也可以實現一個或多個接口,需要的朋友可以參考下 ...

    junjie3542019-12-16
  • JAVA教程學習在一臺新電腦上配置JAVA開發環境

    學習在一臺新電腦上配置JAVA開發環境

    本文主要介紹了如何在一臺新電腦上配置JAVA開發環境,每一個步驟都有對應的截圖和文字說明,需要的朋友可以參考下 ...

    lijiao4922019-12-30
  • JAVA教程java 格式化輸出數字的方法

    java 格式化輸出數字的方法

    在實際工作中,常常需要設定數字的輸出格式,如以百分比的形式輸出,或者設定小數位數等,現稍微總結如下 ...

    java教程網1462019-11-01
  • JAVA教程java實現MD5加密算法的實例代碼

    java實現MD5加密算法的實例代碼

    這篇文章主要介紹了java實現MD5加密算法的實例代碼,有需要的朋友可以參考一下 ...

    java代碼網3042019-10-23
  • JAVA教程java解析xml之dom4j解析xml示例分享

    java解析xml之dom4j解析xml示例分享

    DOM4J具有性能優異、功能強大和極端易用使用的特點,只要懂得DOM基本概念,就可以通過dom4j的api文檔來解析xml,dom4j是一套開源的api。實際項目中,往往選擇...

    java教程網2292019-10-29
  • JAVA教程java編寫貪吃蛇小游戲

    java編寫貪吃蛇小游戲

    貪吃蛇是經典手機游戲,既簡單又耐玩。通過控制蛇頭方向吃蛋,使得蛇變長,從而獲得積分。今天我們就來用java來實現下貪吃蛇小游戲,有需要的小伙伴...

    hebedich4392019-12-14
  • JAVA教程java不通過配置文件初始化logger示例

    java不通過配置文件初始化logger示例

    這篇文章主要介紹了java不通過配置文件初始化logger示例,需要的朋友可以參考下 ...

    Java教程網1922019-11-22
  • JAVA教程Java實現解出世界最難九宮格問題

    Java實現解出世界最難九宮格問題

    這篇文章主要介紹了Java實現解出世界最難九宮格問題,芬蘭數學家因卡拉花費3個月設計出了世界上迄今難度最大的數獨游戲,而且它只有一個答案,本文使用...

    junjie4192019-12-08
主站蜘蛛池模板: 精品国产欧美一区二区三区成人 | 成人性生交大片免费看网站 | 国产精品毛片一区二区三区 | 国产日韩免费视频 | 在线观看国产视频 | 日本女人高潮视频 | 免费福利视频一区二区三区 | 91国内外精品自在线播放 | 久久中文在线观看 | 欧美一区二区日韩一区二区 | 国产中文字幕一区 | 日韩免费视频一区二区 | 丁香六月av | 狠狠躁躁夜夜躁波多野结依 | 国产精品久久久精品 | 日本久久网 | 国产精品一区三区 | 91麻豆精品国产91久久久更新资源速度超快 | 亚洲国产精品久久 | 91高清在线| 先锋影音av在线 | 久久夜色精品国产 | 激情欧美一区二区三区中文字幕 | 涩涩涩久久久成人精品 | 夜夜视频 | 色婷婷精品国产一区二区三区 | 成人免费一区二区三区视频软件 | 亚洲免费在线视频 | 国产视频1区2区 | 99黄色片| 欧美日韩一区二区视频在线观看 | 亚洲高清毛片一区二区 | 欧美日韩精品在线播放 | 91久久久久久久久久 | 日韩视频一区二区三区 | 韩日一区二区三区 | 在线亚洲一区 | 一区二区在线视频 | 亚洲一区中文字幕在线观看 | 自拍在线| 免费午夜电影 |