一般我們說虛函數,它的訪問級別都是public的,用類對象可以直接調用,這樣就可以實現運行時的類型綁定,那如果我們將虛函數私有化會出現什么情況呢?
我們先來看一個非虛函數私有化的例子
class Base { private: void PrintClassName () { cout<<"Base"<<endl; } public: void print() { PrintClassName(); } }; class Derived : public Base { private: void PrintClassName() { cout<<"Derived"<<endl; } };
在main函數里產生一個Derived的對象d,然后調用print()函數,即d.print(),結果輸出的卻是Base,print()函數沒有調用子類的PrintClassName函數,而是調用父類的PrintClassName函數,原來是由於PrintClassName函數不是虛函數之故,所以Base的print()函數調用PrintClassName()函數是在編譯時就已經綁定了,而不是運行期綁定。
下面我們讓PrintClassName()函數變成虛函數再執行,就可以看到輸出的類名為子類的名稱,即Derived。
那么我們有沒有辦法調用私有的虛函數呢?當然是有的,不管公有還是私有,只要是虛函數,它的函數地址都會放在虛函數表vftable中,只要我們找到虛函數表中存放的PrintClassName()函數的地址,我們就可以直接調用,前提是你必須對C++類對象的內存布局要熟悉,代碼如下,這樣也輸出Derived,與前面效果相同
int _tmain(int argc, _TCHAR* argv[]) { Derived d; //d.print(); typedef void (*Fun)(); Fun pFun = NULL; pFun = (Fun)*((int *)(*(int *)&d + 0) + 0); pFun(); getchar(); return 0; }