C++ 中不能被继承的类实现,及从中体现virtual 继承的一个特性
C++ view第一期中給出了如何實現一個不能被繼承的類的方法。
如下:
#ifdef NDEBUG
#define FINAL_CLASS
#else
#define FINAL_CLASS : public virtual Private::NonDerivableHelper? //這里的關鍵是virtual 只要是virtual就不能繼承
#endif
namespace Private
{
class NonDerivableHelper
{
??? protected:
??????? NonDerivableHelper(int) { cout << "a" << endl;} //ADDED CTOR ARGUMENT
};
}
class NonDerivable FINAL_CLASS
{
??? public:
??????? NonDerivable() : NonDerivableHelper(0) {cout << "b" << endl;} //PASS A DUMMY VALUE
};
class A: public NonDerivable
{
??? public:
??????? A() { cout << "c" << endl;}
};
int main(int argc, char *argv[])
{
??? NonDerivable n;
??? A a;
??? return 0;
}
?
?這里A試圖繼承 NonDerivable 類,但是編譯器會報錯
underivable.cc|55| 錯誤: 對‘Private::NonDerivableHelper::NonDerivableHelper()’的調用沒有匹配的函數
underivable.cc|40| 附注: 備選為: Private::NonDerivableHelper::NonDerivableHelper(int)
underivable.cc|38| 附注:????????? Private::NonDerivableHelper::NonDerivableHelper(const Private::NonDerivableHelper&)
這也就是選擇virtual 繼承的原因,選擇virtual繼承,構造函數時會跳過NonDerivable() : NonDerivableHelper(0) 而直接去調用
NonDerivableHelper(int)
在虛繼承出現的繼承層次中,總是在構造非虛基類之前構造虛基類。參考http://tech.ddvip.com/2009-07/1246614951125275.html。
即使是非virtual繼承也是基類先被構造,但是構造函數的選擇是要由子類決定的,如果子類不提供信息則才調用基類的默認構造函數。
而在這里的virtual繼承使得首先就越過NonDerivable,直接去構造NonDerivableHelper。
?
做個小實驗就清楚了
?
#ifdef NDEBUG
#define FINAL_CLASS
#else
#define FINAL_CLASS : public virtual Private::NonDerivableHelper? //這里的關鍵是virtual 只要是virtual就不能繼承
#endif
namespace Private
{
class NonDerivableHelper
{
??? protected:
??????? NonDerivableHelper(int) { cout << "a" << endl;} //ADDED CTOR ARGUMENT
??????? NonDerivableHelper() { cout << "a!" << endl;}
};
}
class NonDerivable FINAL_CLASS
{
??? public:
??????? NonDerivable() : NonDerivableHelper(0) {cout << "b" << endl;} //PASS A DUMMY VALUE
};
class A: public NonDerivable
{
??? public:
??????? A() { cout << "c" << endl;}
};
int main(int argc, char *argv[])
{
??? //A a;
??? NonDerivable n;
??? A a;
??? return 0;
}
?
運行結果
a
b
a!
b
c
將virtual 去掉
即 #define FINAL_CLASS : public Private::NonDerivableHelper? //這里的關鍵是virtual 只要是virtual就不能繼承
運行結果
a
b
a
b
c
OK, virtual? 繼承和非virtual繼承在構造函數調用上的不同已經很清楚了,a與a!的區別。
?
與50位技術專家面對面20年技術見證,附贈技術全景圖總結
以上是生活随笔為你收集整理的C++ 中不能被继承的类实现,及从中体现virtual 继承的一个特性的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: WCF技术剖析之十一:异步操作在WCF中
- 下一篇: RHEL4- DNS服务(四)DNS的开