利用函數指針實現父類函數調用子類函數
父子類關係
對於繼承關係中的父類和子類,我們可以說子類是父類的一種,子類繼承了父類的屬性和行為。因此,子類可以訪問父類的所有非私有成員。相反,父類一般情況下是不能訪問子類成員的。然而,我們可以通過一些方法間接的實現父類訪問子類,即父類函數訪問子類函數。
方法一
利用多態機制,一個指向子類的父類指針或引用,當調用被子類重寫的虛函數時,實際上調用的是子類函數,這是通過多態的方式來實現父類調用子類,該方法需要一個引用或者指針調用虛函數來實現。如下面所示:
Class Base {
public:
virtual void fun()
{
std::cout << "Base::fun()" << std::endl;
}
};
class Derived : public Base
{
public:
virtual void fun()
{
std::cout << "Derived::fun()" << std::endl;
}
};
Base* base = new Derived;
base->fun(); //該結果輸出為: "Derived::fun()"
上面簡單的例子就是通過多態實現的父類調用子類成員函數,這裡面必須要有虛函數,並且在子類裡面重寫。
方法二
通過函數指針同樣可以實現父類函數訪問子類函數,示例代碼如下:
class Base
{
public:
typedef void (Base::*pfn)();
void CallFunction()
{
if (fn != NULL) {
(this->*fn)(); //調用Derived類中的fun()函數
}
}
void SetPfn(pfn fn)
{
this->fn = fn;
}
private:
pfn fn;
};
class Derived : public Base
{
public:
void Init()
{
SetPfn((pfn)&Derived::fun);
}
void fun()
{
std::cout << "Derived::fun be called in Base function!" << std::endl;
}
};
Derived derived;
derived.Init();
derived.CallFunction();
//這裡調用的是父類的成員函數,該函數通過函數指針調用了子類的普通成員函數,
調用結果輸出:"Derived::fun be called in Base function!",即父類成員函數調用了子類成員函數。