成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

C++和java多態的區別

開發 開發工具
C++和java都具多態性,多態性其實就是方法調用的機制,也就是說當在編譯時無法確定一個對象的實際類型時,應當能夠在運行時基于對象的實際類型來決定調用的具體方法。本文主要介紹C++和java多態的區別。

  以前我有個錯誤的觀點:即使在C++java多態性的實現機制可能不同,但它們的表現形式應該相同,也就是說如果代碼結構相同,那么執行結果也應該相同??上屡c愿違,事情并不總是我想象中的那樣子,那么C++和java多態到底有何區別呢?

  首先我們提一下多態性的概念。根據Bjarne Stoustrup的說法,多態性其實就是方法調用的機制,也就是說當在編譯時無法確定一個對象的實際類型時,應當能夠在運行時基于對象的實際類型來決定調用的具體方法(動態綁定)。

  我們先來看一下在C++中的函數調用方式:

  • 普通函數調用:具體調用哪個方法在編譯時間就可以決定(通過查找編譯器的符號表),同時在使用標準過程調用機制基礎上增加一個表示對象身份的指針(this指針)。
  • 虛函數調用:函數調用依賴于對象的實際類型,一般地說,對象的實際類型只能在運行時間才能確定。虛函數一般要有兩個步驟來支持,首先每一個類產生出一堆指向虛函數的指針,放在表格中,這個表格就叫虛函數表(virtual table);然后每一個類對象(class object)會添加一個指向相關虛函數表(virtual table)的指針,通常這個指針叫做vptr。

  在java中又是如何的呢?恩,區別還是滿大的。在java虛擬機中,類實例的引用就是指向一個句柄(handle)的指針,而該句柄(handle)其實是一對指針:其中一個指針指向一張表,該表格包含了對象的方法列表以及一個指向類對象(表示對象類型)的指針;另一個指針指向一塊內存地址,該內存是從java堆中為對象的數據而分配出來的。

  這時,你可能會說,好象差不多嘛,不是都要維護一張函數表嗎?別急,讓我們先看一下例子,這樣你就能更好的理解它們之間的區別到底有多大了。

  下面是C++和java的例子,不看后面的答案,你能夠正確說出它們的執行結果嗎?

  例1:C++

  1.   class Base   
  2.   {   
  3.   public:   
  4.   Base()   
  5.   {   
  6.   init();   
  7.   }   
  8.   virtual ~Base() {}   
  9.   public:   
  10.   virtual void do_init()   
  11.   {   
  12.   init();   
  13.   }   
  14.   protected:   
  15.   virtual void init()   
  16.   {   
  17.   cout << "in Base::init()" << endl;   
  18.   }   
  19.   };   
  20.   class Derived : public Base   
  21.   {   
  22.   public:   
  23.   Derived()   
  24.   {   
  25.   init();   
  26.   }   
  27.   protected:   
  28.   void init()   
  29.   {   
  30.   cout << "in Derived::init()" << endl;   
  31.   }   
  32.   };   
  33.   int main(int argc, char* argv[])   
  34.   {   
  35.   Base* pb;   
  36.   pb = new Derived();   
  37.   delete pb;   
  38.   return 0;   
  39.   }      

  例2:java

  1. class Base   
  2.   {   
  3.   public Base()   
  4.   {   
  5.   init();   
  6.   }   
  7.   protected void init()   
  8.   {   
  9.   System.out.println("in Base::init()");   
  10.   }   
  11.   public void do_init()   
  12.   {   
  13.   init();  
  14.    }  
  15.    }   
  16.   class Derived extends Base   
  17.   {   
  18.   public Derived()   
  19.   {   
  20.   init();   
  21.   }   
  22.   protected void init()   
  23.   {   
  24.   System.out.println("in Derived::init()");  
  25.   }   
  26.   }   
  27.   public class Test   
  28.   {   
  29.   public static void main(String[] args)   
  30.   {   
  31.   Base base = new Derived();   
  32.   }   
  33.   }      

  例1的執行結果是:

  1. in Base::init()   
  2. in Derived::init()      

  例2的執行結果是:

  1. in Derived::init()   
  2. in Derived::init()  
  3.      

  看了結果后,你是馬上頓悟呢抑或是處于疑惑中呢?ok,我們來分析一下兩個例子的執行過程。

  首先看一下例1(C++的例子):

  1. Base* pb; 只是聲明,不做什么。

  2. pb = new Derived();

  1) 調用new操作符,分配內存。

  2) 調用基類(本例中是Base)的構造函數

  3) 在基類的構造函數中調用init(),執行程序首先判斷出當前對象的實際類型是Base(Derived還沒構造出來,當然不會是Derived),所以這里調用的是Base::init()。

  4) 調用派生類(本例中是Derived)的構造函數,在這里同樣要調用init(),執行程序判斷出當前對象的實際類型是Derived,調用Derived::init()。

  3. delete pb; 無關緊要。

  例2(java的例子)的執行過程:

  1. Base base = new Derived();

  1) 分配內存。

  2) 調用基類(本例中是Base)的構造函數

  3) 在基類的構造函數中調用init(),執行程序首先判斷出當前對象的實際類型是Derived(對,Derived已經構造出來,它的函數表當然也已經確定了)所以這里調用的是Derived::init()。

  4) 調用派生類(本例中是Derived)的構造函數,在這里同樣要調用init(),執行程序判斷出當前對象的實際類型是Derived,調用Derived::init()。

  明白了吧。java中的類對象在構造前(調用構造函數之前)就已經存在了,其函數表和對象類型也已經確定了,就是說還沒有出生就已經存在了。而C++中只有在構造完畢后(所有的構造函數都被成功調用)才存在,其函數表和對象的實際類型才會確定。所以這兩個例子的執行結果會不一樣。當然,構造完畢后,C++與java的表現就都一樣了,例如你調用Derived::do_init()的話,其執行結果是:

  1.   in Derived::init() 

  個人認為,java中的多態實現機制沒有C++中的好。還是以例子說明吧:

  例子3:C++

  1.   class Base  
  2.   {  
  3.   public:  
  4.   Base()  
  5.   {  
  6.   init();  
  7.   }  
  8.   virtual ~Base() {}  
  9.   protected:  
  10.   int value;  
  11.   virtual void init()  
  12.   {  
  13.   value = 100;  
  14.   }  
  15.   };  
  16.   class Derived : public Base  
  17.   {  
  18.   public:  
  19.   Derived()  
  20.   {  
  21.   init();  
  22.   }  
  23.   protected:  
  24.   void init()  
  25.   {  
  26.   cout << "value = " << value << endl;  
  27.   // 做一些額外的初始化工作  
  28.   }  
  29.   };  
  30.   int main(int argc, char* argv[])  
  31.   {  
  32.   Base* pb;  
  33.   pb = new Derived();  
  34.   delete pb;  
  35.   return 0;  
  36.   } 

  例4:java

  1.   class Base   
  2.   {   
  3.   public Base()   
  4.   {   
  5.   init();   
  6.   }   
  7.   protected int value;   
  8.   protected void init()   
  9.   {   
  10.   value = 100;   
  11.   }   
  12.   }   
  13.   class Derived extends Base   
  14.   {   
  15.   public Derived()   
  16.   {   
  17.   init();   
  18.   }   
  19.   protected void init()   
  20.   {   
  21.   System.out.println("value = " + value);   
  22.   // 做一些額外的初始化工作   
  23.   }   
  24.   }   
  25.   public class Test   
  26.   {   
  27.   public static void main(String[] args)   
  28.   {   
  29.   Base base = new Derived();   
  30.   }   
  31.   }  

  例3的執行結果是:

  1.   value = 10 

  例4的執行結果是:

  1.   value = 0 
  2.   value = 0 

  從以上結果可以看出,java例子中應該被初始化的值(這里是value)沒有被初始化,派生類根本不能重用基類的初始化函數。試問,如果初始化要在構造時完成,并且初始化邏輯比較復雜,派生類也需要額外的初始化,派生類是不是需要重新實現基類的初始化函數呢?這樣的面向對象方法好不好呢?歡迎大家討論。

【編輯推薦】

  1. 實例演示C++多態的實現過程
  2. C++多態實現方法探討
  3. C++多態性基本概念講述
  4. 深入理解Java多態性
  5. 對Java程序多態性支持的改進
  6. 深入Java核心 Java中多態的實現機制
責任編輯:韓亞珊 來源: 天極網開發頻道
相關推薦

2011-04-12 10:40:04

C++多態

2011-04-11 09:43:25

C++C

2011-07-15 00:47:13

C++多態

2011-06-21 15:00:07

JAVAC++

2011-12-25 15:35:05

ibmdwJavaC++

2024-04-29 07:48:04

C++FinalOverride

2010-01-28 14:38:36

C++和C#、Java

2020-06-17 12:22:44

C覆蓋重載

2022-07-01 11:56:54

C語言C++編程語言

2010-02-03 10:50:33

C++多態

2010-01-08 16:52:57

C++和C#

2010-11-22 16:01:08

C++多態

2010-01-28 15:22:12

C++嵌套類

2024-01-23 10:13:57

C++虛函數

2024-04-22 13:22:00

虛函數象編程C++

2011-07-13 18:00:51

CC++VC

2024-06-28 10:04:09

2009-10-22 09:17:16

C++ CLR

2024-02-26 12:13:32

C++開發編程

2010-02-05 16:07:52

C++多態覆蓋
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美性受xxxx | 国产精品一区久久久久 | 日韩不卡视频在线 | 精品一区二区三区av | 青草久久免费视频 | 日韩精品国产精品 | 免费同性女女aaa免费网站 | 久草电影网 | 久久久久国产一区二区三区四区 | 国产成人91视频 | 免费在线观看成年人视频 | 91亚洲精| 亚州精品天堂中文字幕 | 日韩av免费在线观看 | 亚洲视频1区 | 日韩av高清在线 | 欧美亚洲一区二区三区 | 国产伦一区二区三区视频 | 亚洲精品视频在线 | 中文字幕乱码一区二区三区 | 欧美 日韩 国产 成人 在线 | 成人三级av | 国产欧美在线播放 | 欧美日韩电影一区 | 男人的天堂久久 | 伊人一区 | 欧美精品一区二区三区在线四季 | 欧美精品在线观看 | 久久久精品影院 | 中文在线播放 | 一区二区三区免费观看 | 精品久久久久久 | 一区二区三区免费在线观看 | a级大片免费观看 | 福利网站导航 | 久久久久电影 | 亚洲久久一区 | 国产一区不卡 | 亚洲一区二区网站 | 久久精品国产一区二区电影 | 亚洲国产18|