明輝手游網(wǎng)中心:是一個免費(fèi)提供流行視頻軟件教程、在線學(xué)習(xí)分享的學(xué)習(xí)平臺!

<>


標(biāo)簽:運(yùn)行時完成Java的多態(tài)性 

運(yùn)行時完成Java的多態(tài)性

[摘要]運(yùn)行時多態(tài)性是面向?qū)ο蟪绦蛟O(shè)計代碼重用的一個最強(qiáng)大機(jī)制,動態(tài)性的概念也可以被說成“一個接口,多個方法”。Java實(shí)現(xiàn)運(yùn)行時多態(tài)性的基礎(chǔ)是動態(tài)方法調(diào)度,它是一種在運(yùn)行時而不是在編譯期調(diào)用重載方法的機(jī)制,下面就繼承和接口實(shí)現(xiàn)兩方面談?wù)刯ava運(yùn)行時多態(tài)性的實(shí)現(xiàn)。   一、通過繼承中超類對象引用變量引用...
運(yùn)行時多態(tài)性是面向?qū)ο蟪绦蛟O(shè)計代碼重用的一個最強(qiáng)大機(jī)制,動態(tài)性的概念也可以被說成“一個接口,多個方法”。Java實(shí)現(xiàn)運(yùn)行時多態(tài)性的基礎(chǔ)是動態(tài)方法調(diào)度,它是一種在運(yùn)行時而不是在編譯期調(diào)用重載方法的機(jī)制,下面就繼承和接口實(shí)現(xiàn)兩方面談?wù)刯ava運(yùn)行時多態(tài)性的實(shí)現(xiàn)。

  一、通過繼承中超類對象引用變量引用子類對象來實(shí)現(xiàn)

  舉例說明:

  //定義超類superA
  class superA
  {
  int i = 100;
  void fun()
  {
  System.out.println(“This is superA”);
  }
  }
  //定義superA的子類subB
  class subB extends superA
  {
  int m = 1;
  void fun()
  {
  System.out.println(“This is subB”);
  }
  }
  //定義superA的子類subC
  class subC extends superA
  {
  int n = 1;
  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(); (1)
  a=c;
  a.fun(); (2)
  }
  }

  運(yùn)行結(jié)果為:

  This is subB
  This is subC

  上述代碼中subB和subC是超類superA的子類,我們在類Test中聲明了3個引用變量a, b, c,通過將子類對象引用賦值給超類對象引用變量來實(shí)現(xiàn)動態(tài)方法調(diào)用。也許有人會問:“為什么(1)和(2)不輸出:This is superA”。java 的這種機(jī)制遵循一個原則:當(dāng)超類對象引用變量引用子類對象時,被引用對象的類型而不是引用變量的類型決定了調(diào)用誰的成員方法,但是這個被調(diào)用的方法必須是在超類中定義過的,也就是說被子類覆蓋的方法。

  所以,不要被上例中(1)和(2)所迷惑,雖然寫成a.fun(),但是由于(1)中的a被b賦值,指向了子類subB的一個實(shí)例,因而(1)所調(diào)用的fun()實(shí)際上是子類subB的成員方法fun(),它覆蓋了超類superA的成員方法fun();同樣(2)調(diào)用的是子類subC的成員方法fun()。

  另外,如果子類繼承的超類是一個抽象類,雖然抽象類不能通過new操作符實(shí)例化,但是可以創(chuàng)建抽象類的對象引用指向子類對象,以實(shí)現(xiàn)運(yùn)行時多態(tài)性。具體的實(shí)現(xiàn)方法同上例。

  不過,抽象類的子類必須覆蓋實(shí)現(xiàn)超類中的所有的抽象方法,否則子類必須被abstract修飾符修飾,當(dāng)然也就不能被實(shí)例化了。