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)部類默認(rèn)使用外部類的模板參數(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

