C++的靜態(tài)聯(lián)編和動(dòng)態(tài)聯(lián)編
最近在看析構(gòu)函數(shù)的內(nèi)容,看到一些講的比較好的文章,這里我也有了一些我自己的體會(huì),在這里一并記錄一下。
聯(lián)編是指一個(gè)計(jì)算機(jī)程序自身彼此關(guān)聯(lián)的過程,在這個(gè)聯(lián)編過程中,需要確定程序中的 操作調(diào)用(函數(shù)調(diào)用) 與 執(zhí)行該操作(函數(shù)) 的代碼段之間的映射關(guān)系。
意思就是這個(gè)函數(shù)的實(shí)現(xiàn)有多種,聯(lián)編就是把調(diào)用和對應(yīng)的實(shí)現(xiàn)進(jìn)行映射的操作。
按照聯(lián)編進(jìn)行的階段不同,可分為靜態(tài)聯(lián)編和動(dòng)態(tài)聯(lián)編。
靜態(tài)聯(lián)編
靜態(tài)聯(lián)編工作是在程序編譯連接階段進(jìn)行的,這種聯(lián)編又稱為早期聯(lián)編,因?yàn)檫@種聯(lián)編實(shí)在 程序開始運(yùn)行之前 完成的。在程序編譯階段進(jìn)行的這種聯(lián)編在編譯時(shí)就解決了程序的操作調(diào)用與執(zhí)行該操作代碼間的關(guān)系。
動(dòng)態(tài)聯(lián)編
編譯程序在編譯階段并不能確切地指導(dǎo)將要調(diào)用的函數(shù),只有在程序執(zhí)行時(shí)才能確定將要調(diào)用的函數(shù),為此要確切地指導(dǎo)將要調(diào)用的函數(shù),要求聯(lián)編工作在程序運(yùn)行時(shí)進(jìn)行,這種在 程序運(yùn)行時(shí)進(jìn)行的 聯(lián)編工作被稱為動(dòng)態(tài)聯(lián)編。 C++中,動(dòng)態(tài)聯(lián)編是在虛函數(shù)的支持下實(shí)現(xiàn)的 。
靜態(tài)聯(lián)編和動(dòng)態(tài)聯(lián)編都是屬于多態(tài)性的,他們在不同的階段對不同的實(shí)現(xiàn)進(jìn)行不同的選擇。
動(dòng)態(tài)聯(lián)編需要虛函數(shù)的支持,這是因?yàn)樘摵瘮?shù)的工作原理決定的,而正是因?yàn)槭褂昧颂摵瘮?shù)來實(shí)現(xiàn)動(dòng)態(tài)聯(lián)編,也讓動(dòng)態(tài)聯(lián)編的效率略低于靜態(tài)聯(lián)編。通常,編譯器處理虛函數(shù)的方法是: 給每個(gè)對象添加一個(gè)隱藏成員,隱藏成員保存了一個(gè)指向函數(shù)地址數(shù)組的指針 ,這個(gè)數(shù)組就是虛函數(shù)表(virtual function table, vtbl)。虛函數(shù)表中存儲了為類對象進(jìn)行聲明的虛函數(shù)的地址,調(diào)用虛函數(shù)時(shí),程序?qū)⒉榭创鎯υ趯ο笾械膙tbl地址,然后轉(zhuǎn)向相應(yīng)的函數(shù)地址表,如果使用類聲明中定義的第一個(gè)虛函數(shù),則程序?qū)⑹褂脭?shù)組中的第一個(gè)函數(shù)地址,并執(zhí)行具有該地址的函數(shù),如果使用類聲明中的第三個(gè)虛函數(shù),程序?qū)⑹褂玫刂肺粩?shù)組中第三個(gè)元素的函數(shù)。
虛函數(shù)這個(gè)概念是C++的精華之一。遇到虛函數(shù)時(shí)要注意:
定義一個(gè)函數(shù)為虛函數(shù),不代表函數(shù)為不被實(shí)現(xiàn)的函數(shù)(可以有自己的實(shí)現(xiàn))
定義他位虛函數(shù)是為了允許用基類的指針來調(diào)用子類的這個(gè)函數(shù)(提供了基類調(diào)用子類函數(shù)的方式)
定義一個(gè)函數(shù)為純虛函數(shù),才代表函數(shù)沒有被實(shí)現(xiàn)(聲明后面接=0 virtual func() = 0 此時(shí)派生類必須要實(shí)現(xiàn)此虛函數(shù))
具有純虛函數(shù)的類是 抽象類 ,不能用于生成對象(即不能實(shí)例化),只能派生,他派生的類如果沒有實(shí)現(xiàn)純虛函數(shù),那么他的派生類還是抽象函數(shù)。
虛析構(gòu)函數(shù)
虛析構(gòu)函數(shù)顧名思義就是將析構(gòu)函數(shù)定義為虛函數(shù)。如果我們在派生中分配了內(nèi)存空間,但是基類的析構(gòu)函數(shù)不是虛析構(gòu)函數(shù),就會(huì)發(fā)生內(nèi)存泄漏。先看一個(gè)例子
#include <iostream> using namespace std; class Base { public: Base(){ data = new char[10];} ~Base(){ cout << "destroying Base data[]\n";delete []data;} private: char *data; }; class Derive: public Base { public: Derive(){ D_data = new char[10];} ~Derive(){ cout << "destroying Derive data[]\n";delete []D_data;} private: char *D_data; }; int main() { Base *basePtr = new Derive(); delete basePtr; return 0; }
輸出結(jié)果:
$ ./a.out destroying Base data[]
在這個(gè)例子中,派生類的析構(gòu)函數(shù)并沒有被調(diào)用,這在大的項(xiàng)目中就是一個(gè)災(zāi)難。究其原因是我們在main函數(shù)中定義了一個(gè)Base的指針,當(dāng)我們delete一個(gè)動(dòng)態(tài)分配的Base指針時(shí),Base指針此時(shí)卻指向了Derive類型的對象,但編譯器還是按照Base類型調(diào)用了析構(gòu)函數(shù),沒有執(zhí)行Derive類型的虛析構(gòu)函數(shù)。修改Base類的析構(gòu)函數(shù)為虛析構(gòu)函數(shù)即可以確保執(zhí)行正確的析構(gòu)函數(shù)版本。
最后總結(jié)一下關(guān)于虛函數(shù)的一些常見問題:
- 虛函數(shù)是動(dòng)態(tài)綁定的,也就是說,使用虛函數(shù)的指針和引用能夠正確找到實(shí)際類的對應(yīng)函數(shù),而不是執(zhí)行定義類的函數(shù),這就是虛函數(shù)的基本功能。
- 構(gòu)造函數(shù)不能是虛函數(shù)。而且,在構(gòu)造函數(shù)中調(diào)用虛函數(shù),實(shí)際執(zhí)行的是父類的對應(yīng)函數(shù),因?yàn)樽约哼€么有構(gòu)造好,多態(tài)此時(shí)是被disable的。
- 析構(gòu)函數(shù)可以是虛函數(shù),而且,在一個(gè)復(fù)雜類結(jié)構(gòu)中,這往往是必須的。
- 將基類中的一個(gè)函數(shù)定義為純虛函數(shù),實(shí)際上是將這個(gè)類定義位抽象類,不能實(shí)例化對象。
- 純虛函數(shù)通常沒有定義體,但也可以擁有。(如果Base的析構(gòu)函數(shù)為純虛函數(shù),那么在類外定義Base::~Base(){…}的方式來定義其定義體)
- 析構(gòu)函數(shù)可以是純虛的,但純虛析構(gòu)函數(shù)必須有定義體,因?yàn)槲鰳?gòu)函數(shù)的調(diào)用是在子類中隱含的。
- 非純的虛函數(shù)必須有定義體,不然是一個(gè)錯(cuò)誤。
- 派生類的override虛函數(shù)定義必須和父類完全一致,除了一個(gè)特例,如果父類返回值是一個(gè)指針或引用,子類override時(shí)可以返回這個(gè)指針(或引用)的派生。如在Base中定義了virtual Base clone();在Derive中可以定義virtual Derive clone()。
上一篇:設(shè)計(jì)模式中的備忘錄模式解析及相關(guān)C++實(shí)例應(yīng)用
欄 目:C語言
下一篇:簡單了解設(shè)計(jì)模式中的裝飾者模式及C++版代碼實(shí)現(xiàn)
本文標(biāo)題:C++的靜態(tài)聯(lián)編和動(dòng)態(tài)聯(lián)編
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/2455.html
您可能感興趣的文章
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)式函數(shù)庫
- 04-02c語言中對數(shù)函數(shù)的表達(dá)式 c語言中對數(shù)怎么表達(dá)
- 04-02c語言沒有round函數(shù) round c語言
- 04-02C語言中怎么打出三角函數(shù) c語言中怎么打出三角函數(shù)的值
- 01-10c語言求1+2+...+n的解決方法
- 01-10求子數(shù)組最大和的解決方法詳解
- 01-10深入理解約瑟夫環(huán)的數(shù)學(xué)優(yōu)化方法
- 01-10深入二叉樹兩個(gè)結(jié)點(diǎn)的最低共同父結(jié)點(diǎn)的詳解
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)- 解析最少換車次數(shù)的問題詳解
- 01-10c語言 跳臺階問題的解決方法


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達(dá)式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10delphi制作wav文件的方法
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 08-05織夢dedecms什么時(shí)候用欄目交叉功能?
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 04-02jquery與jsp,用jquery