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

Beginning to 編程

VC++ 方面編程文章

 

C++的多態(tài)性實現機制剖析 /轉孫鑫


 

――即VC++視頻第三課this指針詳細說明

作者:孫鑫 時間:2006年1月12日星期四

要更好地理解C++的多態(tài)性,我們需要弄清楚函數覆蓋的調用機制,因此,首先我們介紹一下函數的覆蓋。

1.   函數的覆蓋

我們先看一個例子:

1- 1

#include <iostream.h>

class animal

{

public:

       void sleep()

       {

              cout<<"animal sleep"<<endl;

       }

       void breathe()

       {

              cout<<"animal breathe"<<endl;

       }

};

class fish:public animal

{

public:

       void breathe()

       {

              cout<<"fish bubble"<<endl;

       }

};

void main()

{

       fish fh;

       animal *pAn=&fh;

       pAn->breathe();

}

       注意,在例1-1的程序中沒有定義虛函數。考慮一下例1-1的程序執(zhí)行的結果是什么?

       答案是輸出:animal breathe

       在類fish中重寫了breathe()函數,我們可以稱為函數的覆蓋。在main()函數中首先定義了一個fish對象fh,接著定義了一個指向animal的指針變量pAn,將fh的地址賦給了指針變量pAn,然后利用該變量調用pAn->breathe()。許多學員往往將這種情況和C++的多態(tài)性搞混淆,認為fh實際上是fish類的對象,應該是調用fish類的breathe(),輸出“fish bubble”,然后結果卻不是這樣。下面我們從兩個方面來講述原因。

1、  編譯的角度

C++編譯器在編譯的時候,要確定每個對象調用的函數的地址,這稱為早期綁定(early binding),當我們將fish類的對象fh的地址賦給pAn時,C++編譯器進行了類型轉換,此時C++編譯器認為變量pAn保存就是animal對象的地址。當在main()函數中執(zhí)行pAn->breathe()時,調用的當然就是animal對象的breathe函數。

2、  內存模型的角度

我們給出了fish對象內存模型,如下圖所示:







圖1- 1 fish類對象的內存模型

我們構造fish類的對象時,首先要調用animal類的構造函數去構造animal類的對象,然后才調用fish類的構造函數完成自身部分的構造,從而拼接出一個完整的fish對象。當我們將fish類的對象轉換為animal類型時,該對象就被認為是原對象整個內存模型的上半部分,也就是圖1-1中的“animal的對象所占內存”。那么當我們利用類型轉換后的對象指針去調用它的方法時,當然也就是調用它所在的內存中的方法。因此,出現圖2.13所示的結果,也就順理成章了。

2.   多態(tài)性和虛函數

正如很多學員所想,在例1-1的程序中,我們知道pAn實際指向的是fish類的對象,我們希望輸出的結果是魚的呼吸方法,即調用fish類的breathe方法。這個時候,就該輪到虛函數登場了。

前面輸出的結果是因為編譯器在編譯的時候,就已經確定了對象調用的函數的地址,要解決這個問題就要使用遲綁定(late binding)技術。當編譯器使用遲綁定時,就會在運行時再去確定對象的類型以及正確的調用函數。而要讓編譯器采用遲綁定,就要在基類中聲明函數時使用virtual關鍵字(注意,這是必須的,很多學員就是因為沒有使用虛函數而寫出很多錯誤的例子),這樣的函數我們稱為虛函數。一旦某個函數在基類中聲明為virtual,那么在所有的派生類中該函數都是virtual,而不需要再顯示的聲明為virtual

下面修改例1-1的代碼,將animal類中的breathe()函數聲明為virtual,如下:

1- 2

#include <iostream.h>

class animal

{

public:

       void sleep()

       {

              cout<<"animal sleep"<<endl;

       }

       virtual void breathe()

       {

              cout<<"animal breathe"<<endl;

       }

};

class fish:public animal

{

public:

       void breathe()

       {

              cout<<"fish bubble"<<endl;

       }

};

void main()

{

       fish fh;

       animal *pAn=&fh;

       pAn->breathe();

}

大家可以再次運行這個程序,你會發(fā)現結果是“fish bubble”,也就是根據對象的類型調用了正確的函數。

那么當我們將breathe()聲明為virtual時,在背后發(fā)生了什么呢?

編譯器在編譯的時候,發(fā)現animal類中有虛函數,此時編譯器會為每個包含虛函數的類創(chuàng)建一個虛表(即vtable),該表是一個一維數組,在這個數組中存放每個虛函數的地址。對于例1-2的程序,animal和fish類都包含了一個虛函數breathe(),因此編譯器會為這兩個類都建立一個虛表,如下圖所示:

圖1- 2 animal類和fish類的虛表

       那么如何定位虛表呢?編譯器另外還為每個類提供了一個虛表指針(即vptr),這個指針指向了對象的虛表。在程序運行時,根據對象的類型去初始化vptr,從而讓vptr正確的指向所屬類的虛表,從而在調用虛函數時,就能夠找到正確的函數。對于例1-2的程序,由于pAn實際指向的對象類型是fish,因此vptr指向的fish類的vtable,當調用pAn->breathe()時,根據虛表中的函數地址找到的就是fish類的breathe()函數。

正是由于每個對象調用的虛函數都是通過虛表指針來索引的,也就決定了虛表指針的正確初始化是非常重要的。換句話說,在虛表指針沒有正確初始化之前,我們不能夠去調用虛函數。那么虛表指針在什么時候,或者說在什么地方初始化呢?

答案是在構造函數中進行虛表的創(chuàng)建和虛表指針的初始化。還記得構造函數的調用順序嗎,在構造子類對象時,要先調用父類的構造函數,此時編譯器只“看到了”父類,并不知道后面是否后還有繼承者,它初始化父類的虛表指針,該虛表指針指向父類的虛表。當執(zhí)行子類的構造函數時,子類的虛表指針被初始化,指向自身的虛表。對于例2-2的程序來說,當fish類的fh對象構造完畢后,其內部的虛表指針也就被初始化為指向fish類的虛表。在類型轉換后,調用pAn->breathe(),由于pAn實際指向的是fish類的對象,該對象內部的虛表指針指向的是fish類的虛表,因此最終調用的是fish類的breathe()函數。

要注意:對于虛函數調用來說,每一個對象內部都有一個虛表指針,該虛表指針被初始化為本類的虛表。所以在程序中,不管你的對象類型如何轉換,但該對象內部的虛表指針是固定的,所以呢,才能實現動態(tài)的對象函數調用,這就是C++多態(tài)性實現的原理。

總結(基類有虛函數):

1、  每一個類都有虛表。

2、  虛表可以繼承,如果子類沒有重寫虛函數,那么子類虛表中仍然會有該函數的地址,只不過這個地址指向的是基類的虛函數實現。如果基類3個虛函數,那么基類的虛表中就有三項(虛函數地址),派生類也會有虛表,至少有三項,如果重寫了相應的虛函數,那么虛表中的地址就會改變,指向自身的虛函數實現。如果派生類有自己的虛函數,那么虛表中就會添加該項。

3、  派生類的虛表中虛函數地址的排列順序和基類的虛表中虛函數地址排列順序相同。

3.   VC視頻第三課this指針說明

我在論壇的VC教學視頻版面發(fā)了帖子,是模擬MFC類庫的例子寫的,主要是說明在基類的構造函數中保存的this指針是指向子類的,我們在看一下這個例子:

例1- 3

#include <iostream.h>

class base;

base * pbase;

class base

{

public:

       base()

       {

              pbase=this;

              

       }

       virtual void fn()

       {

              cout<<"base"<<endl;

       }

};

class derived:public base

{

       void fn()

       {

              cout<<"derived"<<endl;

       }

};

derived aa;

void main()

{

       pbase->fn();

}

我在base類的構造函數中將this指針保存到pbase全局變量中。在定義全局對象aa,即調用derived aa;時,要調用基類的構造函數,先構造基類的部分,然后是子類的部分,由這兩部分拼接出完整的對象aa。這個this指針指向的當然也就是aa對象,那么我們main()函數中利用pbase調用fn(),因為pbase實際指向的是aa對象,而aa對象內部的虛表指針指向的是自身的虛表,最終調用的當然是derived類中的fn()函數。

在這個例子中,由于我的疏忽,在derived類中聲明fn()函數時,忘了加public關鍵字,導致聲明為了private(默認為private),但通過前面我們所講述的虛函數調用機制,我們也就明白了這個地方并不影響它輸出正確的結果。不知道這算不算C++的一個Bug,因為虛函數的調用是在運行時確定調用哪一個函數,所以編譯器在編譯時,并不知道pbase指向的是aa對象,所以導致這個奇怪現象的發(fā)生。如果你直接用aa對象去調用,由于對象類型是確定的(注意aa是對象變量,不是指針變量),編譯器往往會采用早期綁定,在編譯時確定調用的函數,于是就會發(fā)現fn()是私有的,不能直接調用。:)

許多學員在寫這個例子時,直接在基類的構造函數中調用虛函數,前面已經說了,在調用基類的構造函數時,編譯器只“看到了”父類,并不知道后面是否后還有繼承者,它只是初始化父類的虛表指針,讓該虛表指針指向父類的虛表,所以你看到結果當然不正確。只有在子類的構造函數調用完畢后,整個虛表才構建完畢,此時才能真正應用C++的多態(tài)性。換句話說,我們不要在構造函數中去調用虛函數,當然如果你只是想調用本類的函數,也無所謂。

4.   參考資料:

1、文章《在VC6.0中虛函數的實現方法》,作者:backer ,網址:

http://www.mybole.com.cn/bbs/dispbbs.asp?boardid=4&id=1012&star=1

2、書《C++編程思想》 機械工業(yè)出版社

5.   后記

本想再寫詳細些,發(fā)現時間不夠,總是有很多事情,在加上水平也有限,想想還是以后再說吧。不過我相信,這些內容也能夠幫助大家很好的理解了。也歡迎網友能夠繼續(xù)補充,大家可以鼓動鼓動backer,讓他從匯編的角度再給一個說明,哈哈,別說我說的。


posted on 2006-03-09 10:56 Beginning to 編程 閱讀(378) 評論(0)  編輯 收藏 引用 所屬分類: 程序摘錄

導航

統計

常用鏈接

留言簿(4)

隨筆分類

隨筆檔案

文章檔案

相冊

BlogDev

搜索

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            免费观看日韩av| 国产视频欧美视频| 一区二区欧美在线| 亚洲电影免费观看高清完整版在线观看 | 亚洲男人影院| 亚洲精品在线观看免费| 国产在线观看精品一区二区三区| 久久久精彩视频| 午夜一区不卡| 午夜亚洲福利| 欧美中文字幕第一页| 亚洲综合社区| 久久精品国产在热久久| 久久er99精品| 久热国产精品| 欧美黄网免费在线观看| 亚洲国产精品一区二区第一页| 欧美大片在线影院| 欧美激情综合色| 亚洲成人在线网| 亚洲全部视频| 亚洲视频一区二区在线观看 | 久久成人精品| 欧美一区在线看| 久久一区免费| 欧美性猛交xxxx免费看久久久| 国产精品狠色婷| 激情视频亚洲| 亚洲婷婷综合色高清在线| 欧美尤物巨大精品爽| 欧美韩国在线| 亚洲欧美成人| 欧美高清在线| 国内精品久久久久久久影视麻豆 | 欧美日韩国产成人在线观看| 国产精品乱码一区二三区小蝌蚪| 国内成人在线| 亚洲欧美日韩成人| 久久久91精品| 亚洲美女视频| 久久偷窥视频| 国产欧美精品一区二区三区介绍| 亚洲日本黄色| 久久综合久久综合久久| 中文精品一区二区三区| 美女诱惑黄网站一区| 国产欧美日韩综合一区在线播放| 日韩视频中文字幕| 欧美成人国产| 欧美在线影院在线视频| 国产精品成人国产乱一区| 亚洲欧洲中文日韩久久av乱码| 久久爱www.| 亚洲一二区在线| 亚洲日本理论电影| 国产亚洲欧美色| 国产日韩欧美视频| 欧美成人午夜激情在线| 久久婷婷丁香| 亚洲精品午夜精品| 欧美在线视频全部完| 国产精品自在线| 亚洲乱码国产乱码精品精可以看 | 欧美成人免费网站| 亚洲视频第一页| 久久在线免费观看| 久久久久免费| 国产精品久久久久久av下载红粉 | 国产麻豆成人精品| 中文高清一区| 亚洲精品在线观看免费| 免费成人高清视频| 1204国产成人精品视频| 久久综合给合久久狠狠色| 欧美怡红院视频| 国产香蕉久久精品综合网| 欧美一区二区三区精品| 亚洲一区二区黄| 国产一区二区观看| 另类成人小视频在线| 欧美a级一区二区| 亚洲美洲欧洲综合国产一区| 亚洲乱码国产乱码精品精98午夜| 欧美日韩国产影片| 在线综合亚洲| 亚洲天堂黄色| 亚洲国产精品专区久久| 亚洲人成啪啪网站| 国产精品久久久久久户外露出| 久久一区二区三区四区五区| 嫩草国产精品入口| 亚洲女女女同性video| 红桃av永久久久| 性欧美超级视频| 久久精品亚洲| 国内精品久久久久影院优| 欧美韩国一区| 亚洲无亚洲人成网站77777| 国产精品国产自产拍高清av王其| 亚洲欧洲日韩综合二区| 欧美金8天国| 午夜精品亚洲| 久久香蕉国产线看观看av| 国产麻豆综合| 国产精品裸体一区二区三区| 国产亚洲观看| 免费毛片一区二区三区久久久| 欧美精品日本| 久久九九精品99国产精品| 欧美高清视频在线| 亚洲专区一区二区三区| 久久亚洲春色中文字幕久久久| 亚洲视频香蕉人妖| 亚洲欧美日韩视频二区| 亚洲精品一级| 久久久久五月天| 在线一区二区三区四区五区| 久久爱另类一区二区小说| 亚洲精品综合精品自拍| 久久久99久久精品女同性| 一本色道久久| 欧美国产日产韩国视频| 午夜精品短视频| 欧美日韩第一区| 免费不卡在线观看av| 国产日韩欧美在线看| 亚洲精品日韩在线| 亚洲人人精品| 正在播放欧美视频| 日韩视频在线免费观看| 久热这里只精品99re8久| 午夜精品国产更新| 国产精品久久久久久久久婷婷| 欧美成人精品福利| 亚洲国产精品久久久久秋霞不卡 | 欧美色精品在线视频| 欧美激情亚洲另类| 国产日韩在线播放| 性18欧美另类| 午夜亚洲福利| 国产精自产拍久久久久久| 亚洲男人av电影| 欧美视频在线观看一区二区| 99在线精品视频在线观看| 亚洲精品激情| 香蕉久久精品日日躁夜夜躁| 久久综合久久综合久久综合| 亚洲自拍偷拍福利| 国产精品久久久爽爽爽麻豆色哟哟| 亚洲无线一线二线三线区别av| 欧美特黄视频| 午夜精品婷婷| 久久精品日产第一区二区| 国际精品欧美精品| 久久综合久色欧美综合狠狠| 久久久久久久91| 亚洲国产成人在线播放| 亚洲国产成人91精品| 99国产精品久久久| 欧美午夜剧场| 亚洲一区欧美一区| 久久手机免费观看| 国产精品久久久久久福利一牛影视| 亚洲自拍偷拍网址| 久久狠狠久久综合桃花| 在线观看日韩av先锋影音电影院| 久久精品国产欧美亚洲人人爽| 欧美激情片在线观看| 国产美女一区| 欧美成人久久| 黄色一区三区| 欧美日韩一区二区三区高清| 亚洲永久视频| 久久久久成人精品免费播放动漫| 亚洲激情自拍| 欧美巨乳在线观看| 久久国产精品高清| 欧美成人综合网站| 欧美一区1区三区3区公司| 国产一区二区高清不卡| 欧美另类在线播放| 欧美一区二区视频免费观看 | 欧美黄免费看| 一区二区三区四区精品| 国产一区二区观看| 欧美国产日韩精品免费观看| 午夜精品电影| 久久岛国电影| 亚洲欧美春色| 99这里只有久久精品视频| 国产欧美一区二区精品婷婷 | 久久人人97超碰精品888| 亚洲精品在线观看视频| 欧美四级电影网站| 欧美成人午夜77777| 亚洲亚洲精品在线观看| 亚洲国产精品一区二区第一页| 销魂美女一区二区三区视频在线| 日韩视频三区| 国产精品vvv|