C++中的模板類繼承和成員訪問問題
C++模板類繼承和成員訪問
c++中聲明一個模板類及子類,在子類中如果需要訪問父類的protected變量,需要使用父類的類作用域限定符,否則會報“identifier not found”錯誤。
例如:
template<typename T> class A { protected: ?? ?int a; }; ? template<typename T> //模板的繼承,基類需要指定模板參數(shù) class B : public A<T> { public: ?? ?void func() ?? ?{ ? ? ? ? //此處必須使用A::a來訪問;否則會找不到a的定義 ? ? ? ? //如果不是模板類,顯然可以直接使用a來訪問 ?? ??? ?cout << A::a << endl; ?? ?} };
模板函數(shù)的內(nèi)部類繼承時,也是類似的。
例如:
template<typename T> class outer { public: ?? ?class innerBase ?? ?{ ?? ?protected: ?? ??? ?T a; ?? ?}; ? ? ? //在模板類內(nèi)部繼承相同范圍內(nèi)的基類,基類能指定模板參數(shù)(內(nèi)部類默認使用外部類的模板參數(shù)) ? ? //如果繼承其他的模板類,則需要指定模板參數(shù);內(nèi)部類也可以定義自己的模板參數(shù) ?? ?class innerDrive : public innerBase ?? ?{ ?? ?public: ?? ??? ?void func() ?? ??? ?{ ? ? ? ? ? ? //此處必須使用innerBase::a(或者使用outer::innerBase::a)來訪問a ? ? ? ? ? ? //否則無法找到a的定義 ?? ??? ??? ?cout << innerBase::a << endl; ?? ??? ?} ?? ?}; };
類、類模板的protected成員(受保護)繼承規(guī)則
如下,為基類模板和派生類模板的繼承層次:
//基類模板 template<typename T> class D0 { protected: int testi; double testdouble; }; //派生類模板 template<typename T> class D1 :public D0 { public: };
編譯時報錯如下:
原因是派生類只能繼承類,而不能繼承類模板,解決措施就是D0后添加模板類型參數(shù),如下:
//基類模板 template<typename T> class D0 { protected: int testi; double testdouble; }; //派生類模板 template<typename T> class D1 :public D0<T> {//修改處,D0后增加了<T> public: };
現(xiàn)在開始在派生類中使用繼承自基類的protected成員:
//基類模板 template<typename T> class D0 { protected: int testi; double testdouble; }; //派生類模板 template<typename T> class D1 :public D0<T> { public: D1():testi(0) { }//添加了一行使用繼承自基類的protected成員的語句 };
編譯時報錯如下:
原因在于,派生類只能訪問具體某個類的protected成員,不能訪問一個模板的protected成員。
testi未指定究竟是繼承自哪個類的protected成員,解決措施措施如下:
(1)如果想訪問的是繼承自D0類的testi成員,則修改如下:
//基類模板 template<typename T> class D0 { protected: int testi; double testdouble; }; //派生類模板 template<typename T> class D1 :public D0<T> { public: D1():D0<int>::testi(0) { }//訪問繼承自D0<int>的protected成員 };
(2)如果是想訪問跟隨實例化D1的類型T,則修改如下:
//基類模板 template<typename T> class D0 { protected: int testi; double testdouble; }; //派生類模板 template<typename T> class D1 :public D0<T> { public: D1():D0<T>::testi(0) { }//如果D1被實例化時為D1<int>,那么就訪問繼承自D0<int>的protected成員 //如果D1被實例化時為D1<doublie>,那么就訪問繼承自D0<double>的protected成員 //如果D1被實例化時為D1<string>,那么就訪問繼承自D0<string>的protected成員 //………………………………………… };
(2.1)也就是說T會跟隨實例化D1模板的類型而變化。
小結(jié)一下:
(1)派生類不能直接使用基類模板的protected成員,必須指定是使用
具體哪個基類的protected成員。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
C++數(shù)據(jù)結(jié)構(gòu)繼承的概念與菱形繼承及虛擬繼承和組合
今天我要給大家介紹C++中更深入的內(nèi)容了。C++這門語言為了使代碼不冗余,做了些什么操作呢?C++的繼承就很好地實現(xiàn)了類層次的代碼復(fù)用,今天我就要來和大家好好聊一聊它了2022-02-02