C++ View第一期: /*此處一定要聲明,否則在NonDerivableHelper類(lèi)中將NonDerivable定義為友元類(lèi)時(shí),會(huì)認(rèn)為是Private中的某個(gè)類(lèi)*/ 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í)基類(lèi)數(shù)據(jù)在子類(lèi)中出現(xiàn)兩次以上,從而引起訪問(wèn)二義的問(wè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的訪問(wèn)是二義的,但如果B, C都虛擬繼承A,編譯器將保證這種情況不會(huì)出現(xiàn)。
類(lèi)的繼承體系中若有類(lèi)成為virtual base class(即與其子類(lèi)之間均通過(guò)virtual方式繼承的話),那么其構(gòu)造函數(shù)的調(diào)用是由最終具體類(lèi)(本例WantToDerive)來(lái)做的。而WantToDerive是經(jīng)由NonDerivable private繼承NonDerivableHelper,所以無(wú)法調(diào)用NonDerivableHelper的構(gòu)造函數(shù)。 若將virtual去掉,則NonDerivableHelper構(gòu)造函數(shù)的調(diào)用是由NonDerivable來(lái)做的,而NonDerivable是NonDerivableHelper的友元,所以可以訪問(wèn)相應(yīng)的構(gòu)造函數(shù)。
從以上解釋我們可以看到,該例子主要使用了C++中virtual繼承方式和友元不能被繼承的兩個(gè)特性。通過(guò)將NonDerivableHelper 的構(gòu)造函數(shù)定義為private,同時(shí)將子類(lèi)NonDerivable聲明為該類(lèi)的友元類(lèi)。因此NonDerivable可以實(shí)例化。但由于NonDerivable是通過(guò)virtual方式繼承的NonDerivableHelper ,因此其子類(lèi)的構(gòu)造函數(shù)調(diào)用NonDerivableHelper 的構(gòu)造函數(shù)時(shí),是直接調(diào)用,而不是通過(guò)NonDerivable其調(diào)用,故出現(xiàn)“不能調(diào)用私有的NonDerivableHelper::NonDerivableHelper()”錯(cuò)誤,從而實(shí)現(xiàn)了NonDerivable不能被繼承的目的。
|