虛函數(shù)—作為面向?qū)ο缶幊痰暮诵奶匦灾唬摵瘮?shù)不僅在代碼中發(fā)揮著重要作用,更是實(shí)現(xiàn)多態(tài)性的關(guān)鍵所在。
在 C++ 中,虛函數(shù)是為了實(shí)現(xiàn)運(yùn)行時(shí)多態(tài)性而設(shè)計(jì)的特殊類型的函數(shù)。通過(guò)在基類中聲明虛函數(shù),并在派生類中進(jìn)行重寫,可以在程序運(yùn)行時(shí)根據(jù)實(shí)際對(duì)象類型確定調(diào)用的函數(shù)版本。這為我們提供了一種靈活的方式來(lái)處理繼承關(guān)系,使得代碼更具可擴(kuò)展性和可維護(hù)性。
虛函數(shù)表(virtual function table,簡(jiǎn)稱 Vtable)是 實(shí)現(xiàn)虛函數(shù)的重要機(jī)制之一。每個(gè)類(包括含有虛函數(shù)的類)都會(huì)生成一個(gè)對(duì)應(yīng)的虛函數(shù)表,其中存儲(chǔ)了該類中所有虛函數(shù)的地址。
當(dāng)對(duì)象被創(chuàng)建時(shí),會(huì)分配一個(gè)指向其類的虛函數(shù)表的指針(虛指針)。通過(guò)這個(gè)指針,程序能夠在運(yùn)行時(shí)確定調(diào)用的函數(shù)版本,實(shí)現(xiàn)了動(dòng)態(tài)綁定。注意與靜態(tài)綁定混淆重載-靜態(tài)綁定(鏈接)。
派生類的虛函數(shù)表包含基類的虛函數(shù)表內(nèi)容,并擴(kuò)展新函數(shù):派生類的虛函數(shù)表通常是在基類的虛函數(shù)表的基礎(chǔ)上進(jìn)行擴(kuò)展的。
示例代碼解釋 讓我們通過(guò)一段簡(jiǎn)單的代碼來(lái)說(shuō)明虛函數(shù)表的工作原理:
#include <iostream>class Base {public: virtual void func1() { std::cout << "Base::func1()" << std::endl; } virtual void func2() { std::cout << "Base::func2()" << std::endl; }};class Derived : public Base {public: void func1() override { std::cout << "Derived::func1()" << std::endl; } void func3() { std::cout << "Derived::func3()" << std::endl; }};int main() { Base* ptr = new Derived(); ptr->func1(); // 動(dòng)態(tài)綁定 ptr->func2(); // 動(dòng)態(tài)綁定 delete ptr; return 0;}
在這個(gè)示例中,我們創(chuàng)建了一個(gè)基類 Base 和一個(gè)派生類 Derived,后者重寫了基類中的 func1()。
在 main() 函數(shù)中,我們創(chuàng)建了一個(gè)基類指針指向派生類對(duì)象,并通過(guò)該指針調(diào)用了兩個(gè)虛函數(shù) func1() 和 func2()。由于 func1() 是虛函數(shù),并且對(duì)象是 Derived 類型,所以會(huì)動(dòng)態(tài)綁定到 Derived::func1()。而 func2() 在派生類中沒(méi)有被重寫,所以會(huì)綁定到基類的版本。
先看一個(gè)例子(操作環(huán)境64位系統(tǒng))
//先看空類大小class test {};//只有一個(gè)虛函數(shù)的類大小class test1 { public: virtual void function(){ std::cout << "function()" << std::endl; }};//兩個(gè)虛函數(shù)類的大小class test2 { public: virtual void function1(){ std::cout << "function1()" << std::endl; } virtual void function2(){ std::cout << "function2()" << std::endl; }};int main(){ std::cout<<"sizeof test: "<<sizeof(test)<<std::endl; std::cout<<"sizeof test1: "<<sizeof(test1)<<std::endl; std::cout<<"sizeof test2: "<<sizeof(test2)<<std::endl; return 0;}
類在內(nèi)存中記錄虛函數(shù)是以一個(gè)指針記錄的,并且該指針指向一個(gè)數(shù)組,數(shù)組中裝著的是虛函數(shù)的地址。同時(shí),經(jīng)過(guò)實(shí)驗(yàn),64bit的編譯器下,虛函數(shù)表的指針大小是8字節(jié)。
本文鏈接:http://www.tebozhan.com/showinfo-26-84466-0.html探秘C++虛函數(shù):解密多態(tài)的奧秘
聲明:本網(wǎng)頁(yè)內(nèi)容旨在傳播知識(shí),若有侵權(quán)等問(wèn)題請(qǐng)及時(shí)與本網(wǎng)聯(lián)系,我們將在第一時(shí)間刪除處理。郵件:2376512515@qq.com