C++ View第一期:
/*此處一定要聲明,否則在NonDerivableHelper類中將NonDerivable定義為友元類時(shí),會(huì)認(rèn)為是Private中的某個(gè)類*/ class NonDerivable;
namespace Private{
class NonDerivableHelper
{
NonDerivableHelper() {}
friend class NonDerivable;
};
}
#ifdef NDEBUG
#define FINAL_CLASS
#else
#define FINAL_CLASS : private virtual Private::NonDerivableHelper
#endif
class NonDerivable FINAL_CLASS
{
...
};
初次看到這個(gè)例子時(shí),對(duì)其中
的private virtual繼承方法不是很理解。而這種繼承方式恰恰是該例子的精華部分。
virtual在這里的意思是虛擬繼承。它主要是為了解決多重繼承時(shí)基類數(shù)據(jù)在子類中出現(xiàn)兩次以上,從而引起訪問二義的問題。例
class A{
public:
int i;
};
class B:public A
{};
class C:public A
{};
class D:public B, public C
{}//here D has 2 copy of A::i, one comes from B and another comes from C
在上述情況下,對(duì)C::i的訪問是二義的,但如果B, C都虛擬繼承A,編譯器將保證這種情況不會(huì)出現(xiàn)。
類的繼承體系中若有類成為virtual base class(即與其子類之間均通過virtual方式繼承的話),那么其構(gòu)造函數(shù)的調(diào)用是由最終具體類(本例WantToDerive)來做的。而WantToDerive是經(jīng)由NonDerivable private繼承NonDerivableHelper,所以無法調(diào)用NonDerivableHelper的構(gòu)造函數(shù)。
若將virtual去掉,則NonDerivableHelper構(gòu)造函數(shù)的調(diào)用是由NonDerivable來做的,而NonDerivable是NonDerivableHelper的友元,所以可以訪問相應(yīng)的構(gòu)造函數(shù)。
從以上解釋我們可以看到,該例子主要使用了C++中virtual繼承方式和友元不能被繼承的兩個(gè)特性。通過將NonDerivableHelper 的構(gòu)造函數(shù)定義為private,同時(shí)將子類NonDerivable聲明為該類的友元類。因此NonDerivable可以實(shí)例化。但由于NonDerivable是通過virtual方式繼承的NonDerivableHelper ,因此其子類的構(gòu)造函數(shù)調(diào)用NonDerivableHelper 的構(gòu)造函數(shù)時(shí),是直接調(diào)用,而不是通過NonDerivable其調(diào)用,故出現(xiàn)“不能調(diào)用私有的NonDerivableHelper::NonDerivableHelper()”錯(cuò)誤,從而實(shí)現(xiàn)了NonDerivable不能被繼承的目的。
聯(lián)系客服