青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

posts - 195,  comments - 30,  trackbacks - 0

http://hi.baidu.com/daping_zhang/blog/item/e87163d06c42818fa0ec9cfc.html
多態(Polymorphism)是面向對象的核心概念,本文以C++為例,討論多態的具體實現。C++中多態可以分為基于繼承和虛函數的動態多態以及基于模板的靜態多態,如果沒有特別指明,本文中出現的多態都是指前者,也就是基于繼承和虛函數的動態多態。至于什么是多態,在面向對象中如何使用多態,使用多態的好處等等問題,如果大家感興趣的話,可以找本面向對象的書來看看。
    為了方便說明,下面舉一個簡單的使用多態的例子(From [1] ):

class Shape
{
protected:
  int m_x;    // X coordinate
  int m_y;  // Y coordinate
public:
  // Pure virtual function for drawing
  virtual void Draw() = 0;  

  // A regular virtual function
  virtual void MoveTo(int newX, int newY);

 // Regular method, not overridable.
  void Erase();

  // Constructor for Shape
  Shape(int x, int y); 

 // Virtual destructor for Shape
  virtual ~Shape();
};
// Circle class declaration
class Circle : public Shape
{
private:
   int m_radius;    // Radius of the circle 
public:
   // Override to draw a circle
   virtual void Draw();    

   // Constructor for Circle
   Circle(int x, int y, int radius);

  // Destructor for Circle
   virtual ~Circle();
};
// Shape constructor implementation
Shape::Shape(int x, int y)
{
   m_x = x;
   m_y = y;
}
// Shape destructor implementation
Shape::~Shape()
{
//...
}
 // Circle constructor implementation
Circle::Circle(int x, int y, int radius) : Shape (x, y)
{
   m_radius = radius;
}

// Circle destructor implementation
Circle::~Circle()
{
//...
}

// Circle override of the pure virtual Draw method.
void Circle::Draw()
{
   glib_draw_circle(m_x, m_y, m_radius);
}

main()
{
  // Define a circle with a center at (50,100) and a radius of 25
  Shape *pShape = new Circle(50, 100, 25);

  // Define a circle with a center at (5,5) and a radius of 2
  Circle aCircle(5,5, 2);

  // Various operations on a Circle via a Shape pointer
  //Polymorphism
  pShape->Draw();
  pShape->MoveTo(100, 100);

  pShape->Erase();
  delete pShape;

 // Invoking the Draw method directly
  aCircle.Draw();
}   

     例子中使用到多態的代碼以黑體標出了,它們一個很明顯的特征就是通過一個基類的指針(或者引用)來調用不同子類的方法。
     那么,現在的問題是,這個功能是怎樣實現的呢?我們可以先來大概猜測一下:對于一般的方法調用,到了匯編代碼這一層次的時候,一般都是使用 Call funcaddr 這樣的指令進行調用,其中funcaddr是要調用函數的地址。按理來說,當我使用指針pShape來調用Draw的時候,編譯器應該將Shape::Draw的地址賦給funcaddr,然后Call 指令就可以直接調用Shape::Draw了,這就跟用pShape來調用Shape::Erase一樣。但是,運行結果卻告訴我們,編譯器賦給funcaddr的值卻是Circle::Drawde的值。這就說明,編譯器在對待Draw方法和Erase方法時使用了雙重標準。那么究竟是誰有這么大的法力,使編譯器這個鐵面無私的判官都要另眼相看呢?virtual!!
     
Clever!!正是virtual這個關鍵字一手導演了這一出“乾坤大挪移”的好戲。說道這里,我們先要明確兩個概念:靜態綁定和動態綁定。
    1、靜態綁定(static bingding),也叫早期綁定,簡單來說就是編譯器在編譯期間就明確知道所要調用的方法,并將該方法的地址賦給了Call指令的funcaddr。因此,運行期間直接使用Call指令就可調用到相應的方法。
    2、動態綁定(dynamic binding),也叫晚期綁定,與靜態綁定不同,在編譯期間,編譯器并不能明確知道究竟要調用的是哪一個方法,而這,要知道運行期間使用的具體是哪個對象才能決定。
    好了,有了這兩個概念以后,我們就可以說,virtual的作用就是告訴編譯器:我要進行動態綁定!編譯器當然會尊重你的意見,而且為了完成你這個要求,編譯器還要做很多的事情:編譯器自動在聲明了virtual方法的類中插入一個指針vptr和一個數據結構VTable(vptr用以指向VTable;VTable是一個指針數組,里面存放著函數的地址),并保證二者遵守下面的規則:
    1、VTable中只能存放聲明為virtual的方法,其它方法不能存放在里面。在上面的例子中,Shape的VTable中就只有Draw,MoveTo和~Shape。方法Erase的地址并不能存放在VTable中。此外,如果方法是純虛函數,如 Draw,那么同樣要在VTable中保留相應的位置,但是由于純虛函數沒有函數體,因此該位置中并不存放Draw的地址,而是可以選擇存放一個出錯處理的函數的地址,當該位置被意外調用時,可以用出錯函數進行相應的處理。
    2、派生類的VTalbe中記錄的從基類中繼承下來的虛函數地址的索引號必須跟該虛函數在基類VTable中的索引號保持一致。如在上例中,如果在Shape的VTalbe中,Draw為 1 號, MoveTo 2 號,~Shape為 3 號,那么,不管這些方法在Circle中是按照什么順序定義的,Circle的VTable中都必須保證Draw為 1 號,MoveTo為 2號。至于 3號,這里是~Circle。為什么不是~Shape啊?嘿嘿,忘啦,析構函數不會繼承的。
    3、vptr是由編譯器自動插入生成的,因此編譯器必須負責為其進行初始化。初始化的時間選在對象創建時,而地點就在構造函數中。因此,編譯器必須保證每個類至少有一個構造函數,若沒有,自動為其生成一個默認構造函數。
     4、vptr通常放在對象的起始處,也就是Addr(obj) == Addr(obj.vptr)。
    你看,天下果然沒有免費的午餐,為了實現動態綁定,編譯器要為我們默默干了這么多的臟話累活。如果你想體驗一下編譯器的辛勞,那么可以嘗試用C語言模擬一下上面的行為,【1】中就有這么一個例子。好了,現在萬事具備,只欠東風了。編譯,連接,載入,GO!當程序執行到 pShape->Draw()的時候,上面的設施也開始起作用了。。
    前面已經提到,晚期綁定時之所以不能確定調用哪個函數,是因為具體的對象不確定。好了,當運行到pShape->Draw()時,對象出來了,它由pShape指針標出。我們找到這個對象后,就可以找到它里面的vptr(在對象的起始處),有了vptr后,我們就找到了VTable,調用的函數就在眼前了。。等等,VTable中方法那么多,我究竟使用哪個呢?不用著急,編譯器早已為我們做好了記錄:編譯器在創建VTable時,已經為每個virtual函數安排好了座次,并且把這個索引號記錄了下來。因此,當編譯器解析到pShape->Draw()的時候,它已經悄悄的將函數的名字用索引號來代替了。這時候,我們通過這個索引號就可以在VTable中得到一個函數地址,Call it!
    在這里,我們就體會到為什么會有第二條規定了,通常,我們都是用基類的指針來引用派生類的對象,但是不管具體對象是哪個派生類的,我們都可以使用相同的索引號來取得對應的函數實現。
     現實中有一個例子其實跟這個蠻像的:報警電話有110,119,120(VTable中不同的方法)。不同地方的人撥打不同的號碼所產生的結果都是不一樣的。譬如,在三環外的一個人(具體對象)跟一環內的一個人(另外一個具體對象)打119,最后調用的消防隊肯定是不一樣的,這就是多態了。這是怎么實現的呢,每個人都知道一個報警中心(VTable,里面有三個方法 110,119,120)。如果三環外的一個人需要火警搶險(一個具體對象)時,它就撥打119,但是他肯定不知道最后是哪一個消防隊會出現的。這得有報警中心來決定,報警中心通過這個具體對象(例子中就是具體位置了)以及他說撥打的電話號碼(可以理解成索引號),報警中心可以確定應該調度哪一個消防隊進行搶險(不同的動作)。
     這樣,通過vptr和VTable的幫助,我們就實現了C++的動態綁定。當然,這僅僅是單繼承時的情況,多重繼承的處理要相對復雜一點,下面簡要說一下最簡單的多重繼承的情況,至于虛繼承的情況,有興趣的朋友可以看看 Lippman的《Inside the C++ Object Model》,這里暫時就不展開了。(主要是自己還沒搞清楚,況且現在多重繼承都不怎么使用了,虛繼承應用的機會就更少了)
     首先,我要先說一下多重繼承下對象的內存布局,也就是說該對象是如何存放本身的數據的。

class Cute
{
public:
 int i;
 virtual void cute(){ cout<<"Cute cute"<<endl; }
};
class Pet
{
public:
   int j;
   virtual void say(){ cout<<"Pet say"<<endl;  }
};
class Dog : public Cute,public Pet
{
public:
 int z;
 void cute(){ cout<<"Dog cute"<<endl; }
 void say(){ cout<<"Dog say"<<endl;  }
};

    在上面這個例子中,一個Dog對象在內存中的布局如下所示:                    

Dog

Vptr1

Cute::i

Vptr2

Pet::j

Dog::z


     也就是說,在Dog對象中,會存在兩個vptr,每一個跟所繼承的父類相對應。如果我們要想實現多態,就必須在對象中準確地找到相應的vptr,以調用不同的方法。但是,如果根據單繼承時的邏輯,也就是vptr放在指針指向位置的起始處,那么,要在多重繼承情況下實現,我們必須保證在將一個派生類的指針隱式或者顯式地轉換成一個父類的指針時,得到的結果指向相應派生類數據在Dog對象中的起始位置。幸好,這工作編譯器已經幫我們完成了。上面的例子中,如果Dog向上轉換成Pet的話,編譯器會自動計算Pet數據在Dog對象中的偏移量,該偏移量加上Dog對象的起始位置,就是Pet數據的實際地址了。

int main()
{
 Dog* d = new Dog();
 cout<<"Dog object addr : "<<d<<endl;
 Cute* c = d;
 cout<<"Cute type addr : "<<c<<endl;
 Pet* p = d;
 cout<<"Pet type addr : "<<p<<endl;
 delete d;
}
output:
Dog object addr : 0x3d24b0
Cute type addr : 0x3d24b0
Pet type addr : 0x3d24b8   // 正好指向Dog對象的vptr2處,也就是Pet的數據

      好了,既然編譯器幫我們自動完成了不同父類的地址轉換,我們調用虛函數的過程也就跟單繼承統一起來了:通過具體對象,找到vptr(通常指針的起始位置,因此Cute找到的是vptr1,而Pet找到的是vptr2),通過vptr,我們找到VTable,然后根據編譯時得到的VTable索引號,我們取得相應的函數地址,接著就可以馬上調用了。

      在這里,順便也提一下兩個特殊的方法在多態中的特別之處吧:第一個是構造函數,在構造函數中調用虛函數是不會有多態行為的,例子如下:

class Pet
{
public:
   Pet(){ sayHello(); }
   void say(){ sayHello(); }

   virtual void sayHello()
   {
     cout<<"Pet sayHello"<<endl;
   }
   
};
class Dog : public Pet
{
public:
   Dog(){};
   void sayHello()
   {
     cout<<"Dog sayHello"<<endl;
   }
};
int main()
{
 Pet* p = new Dog();
 p->sayHello();
 delete p;
}
output:
Pet sayHello //直接調用的是Pet的sayHello()
Dog sayHello //多態

     第二個就是析構函數,使用多態的時候,我們經常使用基類的指針來引用派生類的對象,如果是動態創建的,對象使用完后,我們使用delete來釋放對象。但是,如果我們不注意的話,會有意想不到的情況發生。

class Pet
{
public:
   ~Pet(){ cout<<"Pet destructor"<<endl;  }
  //virtual ~Pet(){ cout<<"Pet virtual destructor"<<endl;  }
};
class Dog : public Pet
{
public:
   ~Dog(){ cout<<"Dog destructor"<<endl;};
   //virtual ~Dog(){ cout<<"Dog virtual destructor"<<endl;  }
};
int main()
{
 Pet* p = new Dog();
 delete p;
}
output:
Pet destructor  //糟了,Dog的析構函數沒有調用,memory leak!

如果我們將析構函數改成virtual以后,結果如下
Dog virtual destructor
Pet virtual destructor   // That's OK!

    所以,如果一個類設計用來被繼承的話,那么它的析構函數應該被聲明為virtual的。

Reference:
[1] Comparing C++ and C (Inheritance and Virtual Functions)  
[2] C++對象布局及多態實現的探索 
[3] Multiple inheritance and the this pointer 講述多重繼承下的類型轉換問題
[4] Memory Layout for Multiple and Virtual Inheritance 詳細描述了多重菱形多重繼承下的對象內存布局以及類型轉換 

后記:當我完成了本篇%90的時候,我試圖提交,誰知道登陸太久沒有動作,session超時,讓我重新登陸,然后提交的內容就全部不見了,剩下最開始的%10。。。。當時的心情啊,用呂大哥的話來說就是:尋死的心都有了。

posted on 2011-04-08 22:09 luis 閱讀(393) 評論(0)  編輯 收藏 引用

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


<2011年3月>
272812345
6789101112
13141516171819
20212223242526
272829303112
3456789

常用鏈接

留言簿(3)

隨筆分類

隨筆檔案

文章分類

文章檔案

友情鏈接

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲乱码日产精品bd| 欧美成人69av| 国产精品婷婷| 欧美性色综合| 国产精品久久久久久久久婷婷| 欧美日韩精品免费观看| 欧美日韩免费观看一区=区三区| 麻豆精品网站| 欧美精品久久久久久久久老牛影院 | 在线视频精品| 欧美在线一级视频| 免费日韩av片| 国产精品久久久| 国产一区日韩欧美| 日韩特黄影片| 久久青青草原一区二区| 欧美激情一区二区三区| 一区二区av| 久久久久国产精品人| 欧美日韩久久精品| 一区二区三区我不卡| 中文一区二区在线观看| 久久免费午夜影院| 一个色综合导航| 久久久久99| 国产精品免费网站| 亚洲伦理一区| 麻豆av一区二区三区| 亚洲一区日韩在线| 欧美久色视频| 亚洲欧洲日韩综合二区| 久久久久看片| 亚洲少妇诱惑| 欧美日韩1234| 亚洲激情不卡| 免费h精品视频在线播放| 亚洲一区不卡| 欧美日韩亚洲一区三区 | 国产一区在线看| 亚洲一本大道在线| 欧美国产日产韩国视频| 午夜在线观看免费一区| 欧美午夜剧场| 一区二区三区四区五区精品| 久久婷婷国产麻豆91天堂| 这里只有精品丝袜| 欧美区二区三区| 亚洲日本va在线观看| 久久精品理论片| 亚洲综合丁香| 国产精品人成在线观看免费| 亚洲在线电影| 一区二区三区日韩| 欧美偷拍一区二区| 亚洲一区二区黄| 一区二区欧美国产| 亚洲一区二区三区涩| 亚洲盗摄视频| 午夜久久美女| 国产日韩综合| 欧美在线播放一区二区| 亚洲一区二区免费看| 欧美视频一区二区在线观看| 99精品欧美一区| 一本色道久久综合亚洲91| 国产精品theporn| 性欧美激情精品| 亚洲欧美视频一区| 国内一区二区三区在线视频| 久久免费高清| 欧美91福利在线观看| 亚洲精品乱码久久久久久蜜桃91| 欧美成人一区二区三区| 麻豆成人av| 99这里只有久久精品视频| 亚洲精品日韩激情在线电影| 国产精品国产福利国产秒拍| 久久xxxx| 欧美成人a∨高清免费观看| 一区二区三区|亚洲午夜| 亚洲视频成人| 狠狠色2019综合网| 亚洲人成在线播放网站岛国| 国产精品福利av| 久久综合九九| 欧美日韩国产在线播放| 欧美一级在线播放| 久久亚洲不卡| 亚洲女人天堂av| 久久久亚洲国产天美传媒修理工 | 亚洲欧美资源在线| 久久se精品一区精品二区| 91久久久亚洲精品| 亚洲视频综合| 亚洲第一区在线| 在线综合欧美| 亚洲国产91精品在线观看| 99精品福利视频| 国内精品免费午夜毛片| 日韩视频专区| 在线观看亚洲专区| 亚洲性线免费观看视频成熟| 在线观看一区二区精品视频| 亚洲天堂成人| 99在线|亚洲一区二区| 久久精品99国产精品| 一本色道综合亚洲| 久久免费高清视频| 香蕉视频成人在线观看| 欧美国产在线电影| 久久亚洲综合色一区二区三区| 欧美日本韩国一区| 欧美国产亚洲视频| 国内视频一区| 亚洲综合欧美日韩| 国产精品99久久久久久有的能看| 欧美成人一区二区三区| 国产精品久久久久影院亚瑟| 欧美黄色一区| 国内自拍一区| 性色av一区二区三区在线观看 | 亚洲精品国产拍免费91在线| 国产在线高清精品| 亚洲——在线| 西瓜成人精品人成网站| 欧美日韩国产在线播放网站| 亚洲黄页一区| 亚洲精品一区二区三区在线观看| 久久精品综合一区| 久久久久免费| 一区在线免费观看| 久久久www免费人成黑人精品| 久久国产色av| 国产一区二区三区在线免费观看| 午夜精品一区二区三区在线视| 亚洲午夜电影网| 欧美色图首页| 亚洲女人天堂av| 久久精品99久久香蕉国产色戒| 国产精品久线观看视频| 亚洲一区二区日本| 欧美一区二区精品| 国产日韩亚洲欧美| 欧美在线视频一区二区| 久久综合五月| 在线成人av.com| 男人天堂欧美日韩| 亚洲精品美女在线观看| 亚洲在线观看视频网站| 国产欧美综合在线| 久久亚洲欧洲| 亚洲欧洲美洲综合色网| 亚洲一二三区在线| 国产日韩欧美一区二区| 久久伊人精品天天| 9久草视频在线视频精品| 欧美中在线观看| 亚洲高清网站| 国产精品久久九九| 久久亚洲风情| 亚洲视频电影图片偷拍一区| 久久精品av麻豆的观看方式| 亚洲国产高潮在线观看| 欧美日韩一级视频| 久久国产精彩视频| 亚洲国产专区校园欧美| 亚洲欧美国产77777| 一区二区三区无毛| 国产精品久久久久av| 久久综合色天天久久综合图片| 亚洲日本理论电影| 久久久久一区二区三区| 99视频日韩| 国产亚洲欧美日韩精品| 欧美人交a欧美精品| 久久成人人人人精品欧| 一区二区欧美在线| 欧美激情一区在线| 久久九九热免费视频| 一区二区三区三区在线| 一区在线免费| 国产色综合久久| 欧美手机在线| 欧美freesex8一10精品| 午夜日韩在线| 欧美一级淫片播放口| 亚洲人成在线观看| 国内成人精品2018免费看| 国产精品啊啊啊| 欧美国产高清| 久久久久在线观看| 午夜激情综合网| 亚洲午夜黄色| 亚洲人成高清| 亚洲电影免费在线| 女生裸体视频一区二区三区| 欧美在线观看一二区| 亚洲一区欧美二区| 亚洲午夜精品久久久久久app| 亚洲精品国精品久久99热|