C++ View第一期: /*此處一定要聲明,否則在NonDerivableHelper類中將NonDerivable定義為友元類時,會認(rèn)為是Private中的某個類*/ 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 { ... }; 初次看到這個例子時,對其中
的private virtual繼承方法不是很理解。而這種繼承方式恰恰是該例子的精華部分。 virtual在這里的意思是虛擬繼承。它主要是為了解決多重繼承時基類數(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 在上述情況下,對C::i的訪問是二義的,但如果B, C都虛擬繼承A,編譯器將保證這種情況不會出現(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繼承方式和友元不能被繼承的兩個特性。通過將NonDerivableHelper 的構(gòu)造函數(shù)定義為private,同時將子類NonDerivable聲明為該類的友元類。因此NonDerivable可以實(shí)例化。但由于NonDerivable是通過virtual方式繼承的NonDerivableHelper ,因此其子類的構(gòu)造函數(shù)調(diào)用NonDerivableHelper 的構(gòu)造函數(shù)時,是直接調(diào)用,而不是通過NonDerivable其調(diào)用,故出現(xiàn)“不能調(diào)用私有的NonDerivableHelper::NonDerivableHelper()”錯誤,從而實(shí)現(xiàn)了NonDerivable不能被繼承的目的。
|