C++ 虛函數(shù)專題
虛函數(shù)
基類中使用virtual關鍵字聲明的函數(shù),稱為虛函數(shù)。
虛函數(shù)的實現(xiàn),通過虛函數(shù)表來實現(xiàn)的。即V-table 這個表中有一個類,用于儲存虛函數(shù)的地址。解決其繼承,覆蓋的問題,用于保證其真實反映的函數(shù)。這樣有虛函數(shù)的實例,將會儲存在這個實例的內存中。即用父類的指針,操作子類的時候,通過虛函數(shù)表來實現(xiàn)找尋到父類。
定義下方的一個類
class Base{ public: virtual void f(){ cout << "Base::f" << endl; } virtual void g(){ cout << "Base::g" << endl; } virtual void h(){ cout << "Base::h" << endl; } }
然后通過實例化得到虛函數(shù)表
Base* b = new Base(); //Fun pFun = NULL; cout << "虛函數(shù)表地址 " << (int*)(&(*b)) << endl; // 強制轉換成為指針類型,然后輸出 cout << "虛函數(shù)表 - 第一個函數(shù)地址" << (int*)*(int*)(&(*b)) << endl; // 取虛函數(shù)的地址,然后獲得虛函數(shù)的頭指向的第一個存儲函數(shù)的內存空間。然后獲取第一個函數(shù)的內存地址
然后查看輸出的結果
虛函數(shù)表地址 0xb31268
虛函數(shù)表 - 第一個函數(shù)地址0x4c1490
根據(jù)地址可以看到,指針b指向創(chuàng)建的實例的地址,其首地址儲存著虛函數(shù)表的地址,然后再次通過指針訪問,得到虛函數(shù)表的第一個函數(shù)的指針的地址。
一般繼承,沒有虛函數(shù)覆蓋
在上方的繼承中,子類沒有重載任何父類的函數(shù),那么在虛函數(shù)列表中,就代表著
虛函數(shù)按照聲明的順序放入表中。
父類的虛函數(shù),在子類的虛函數(shù)的前面。
一般繼承,有虛函數(shù)覆蓋
此時內存中地址如下
注意,最重要的一點是,地址上,覆蓋的f()函數(shù),被放置到了父類的f()函數(shù)上。于是就可以有下方的程序
Base *b = new Derive(); b->f();
使用一個類型為Base的指針b指向一個新建的實例Derive(),此時對于指針b指向的虛函數(shù)表中,此時f()的地址被Devieive函數(shù)的地址被覆蓋, 形成如上的虛函數(shù)表。
此時訪問成功
多重繼承 無虛函數(shù)的覆蓋
假設有如上的內容
此時虛函數(shù)表
每個父類都有自己的虛表,子類的成員函數(shù)被放到第一個父類的表中。即第一個父類的表是按照聲明的順序來判斷。
多重繼承 有虛函數(shù)覆蓋
此時表如下
此時父類的被替換了。
這樣就實現(xiàn)了多重繼承,代碼如下
Derive d; Base1 *b1 = &d; Base2 *b2 = &d; Base3 *b3 = &d; b1->f(); b2->f(); b3->f();
此時代碼如上。
純虛函數(shù)
virtual int area() = 0;
實現(xiàn)一個純虛函數(shù),此時可以在派生類中更好的定義純虛函數(shù)。
以上就是C++ 虛函數(shù)專題的詳細內容,更多關于C++ 虛函數(shù)的資料請關注腳本之家其它相關文章!
相關文章
vc++實現(xiàn)的tcp socket客戶端和服務端示例
這篇文章主要介紹了vc++實現(xiàn)的tcp socket客戶端和服務端示例,需要的朋友可以參考下2014-03-03C++ STL priority_queue自定義排序實現(xiàn)方法詳解
這篇文章主要介紹了C++ STL priority_queue自定義排序實現(xiàn)方法詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-03-03C語言數(shù)據(jù)結構實現(xiàn)鏈表去重的實例
這篇文章主要介紹了C語言數(shù)據(jù)結構實現(xiàn)鏈表去重的實例的相關資料,這里提供了題目及實例代碼,需要的朋友可以參考下2017-07-07