C++之private虛函數


一般我們說虛函數,它的訪問級別都是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;
}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM