一般在繼承的時(shí)候, 子類是否能訪問父類的成員變量, 需要分情況討論
1. 同類不同模板
//下面這個(gè)例子跟繼承沒有關(guān)系, 不過先來探討一下
template<typename C>
class A
{
private:
int data; //data本身需要是private的, 這樣只有在該類的對象中才能訪問
public:
template<typename M>
void test(A<M>& a) {
a.data;
}
};
int main() {
A<int> a1;
A<float> a2;
a1.test(a1); //此處調(diào)用a1對象的成員變量, 并傳入一個(gè)A<int>對象, 沒有問題
a1.test(a2); //此處調(diào)用a1對象的成員變量, 傳入一個(gè)A<float>, 此時(shí)編譯器將a1和a2認(rèn)為是兩個(gè)不同的類型, 此處報(bào)錯(cuò)為: 'data' is a private member of 'A<float>', 不過這個(gè)報(bào)錯(cuò)倒是因編譯器而異
}
2. 形參類型與所在類的關(guān)系
2.1 形參類型與所在類類型相同
- 如果成員變量是public和protected的, 則可以直接訪問
- 如果成員變量是private的, 不可訪問
//在Apple的sale這個(gè)方法中, 傳入的形參是Apple類型, 而sale方法本身也在Apple類中, 故此時(shí)形參類型與所在類類型相同
class Fruit {
private:
int name;
protected:
int type;
};
class Apple : public Fruit { //此處的public跟Apple中是否能訪問Fruit的變量無關(guān), 此處的public是標(biāo)明從Fruit處繼承來的變量的屬性
public:
void sale(Apple& apple){
apple.name; //name是private的, 此處會(huì)報(bào)錯(cuò), 無法訪問
apple.type; //type是protected的, 此處可以直接訪問
}
};
2.1 形參類型與所在類類型不同
- 如果成員變量是public的, 則可以直接訪問
- 如果成員變量是private和protected的, 不可訪問
//我們來看一個(gè)極端一點(diǎn)的例子
class Fruit{
protected:
int name;
};
class Apple : public Fruit {//雖然Apple繼承于Fruit, 不過依然無法訪問Fruit中protected的變量
public:
void Method(Fruit& fruit) { //此處形參類型與類類型不同
fruit.name; //此處報(bào)錯(cuò), 因?yàn)閚ame是protected的
}
};
3 訪問控制符
3.1 當(dāng)所在類類型與形參類型不同, 且沒有繼承關(guān)系時(shí)
- 如果繼承時(shí)的訪問控制符是public, 那么僅能訪問父類的public成員
- 如果繼承時(shí)的訪問控制符是protected, 那么父類所有類型成員都無法訪問
- 如果繼承時(shí)的訪問繼承父是private, 那么父類的所有類型成員都無法訪問
//之前提到過在繼承時(shí)的訪問控制符起的功效是將父類中繼承的變量, 按照訪問控制符的類型加載到子類中
class Fruit1 {
private:
int a1;
protected:
int b1;
public:
int c1;
};
class Fruit2 {
private:
int a2;
protected:
int b2;
public:
int c2;
};
class Fruit3 {
private:
int a3;
protected:
int b3;
public:
int c3;
};
class Apple : public Fruit1, protected Fruit2, private Fruit3 {
};
class Banana {
public:
void sale(Apple& apple){
apple.a1; // 此處因?yàn)閥是Apple類而當(dāng)前所在類時(shí)Banana, 報(bào)錯(cuò)
apple.b1; // 因?yàn)槠涫莗rotected對象, 報(bào)錯(cuò)
apple.c1; // 可以訪問
apple.a2; // 報(bào)錯(cuò)
apple.b2; // 報(bào)錯(cuò)
apple.c2; // 報(bào)錯(cuò)
apple.a3; // 報(bào)錯(cuò)
apple.b3; // 報(bào)錯(cuò)
apple.c3; // 報(bào)錯(cuò)
}
};
3.2 當(dāng)所在類類型與形參類型不同, 但有繼承關(guān)系時(shí)
- 如果繼承時(shí)的訪問控制符是public, 那么可以訪問父類的public和protected成員
- 如果繼承時(shí)的訪問控制符是protected, 那么可以訪問父類的public和protected成員
- 如果繼承時(shí)的訪問繼承父是private, 那么父類的所有類型成員都無法訪問
//在每個(gè)Fruit中變量的定義順序都是a,b,c <-> private, protected, public
class Fruit1 {
private:
int a1;
protected:
int b1;
public:
int c1;
};
class Fruit2 {
private:
int a2;
protected:
int b2;
public:
int c2;
};
class Fruit3 {
private:
int a3;
protected:
int b3;
public:
int c3;
};
class Apple : public Fruit1, protected Fruit2, private Fruit3 {
};
class ApplePear:Apple { //Apple與Fruit3的繼承關(guān)系時(shí)private的, ApplePear是無法獲知該繼承關(guān)系, 也就無法繼承Fruit3中的值
public:
void sale(Apple& apple){
apple.a1; // 報(bào)錯(cuò)
apple.b1; // 可以訪問
apple.c1; // 可以訪問
apple.a2; // 報(bào)錯(cuò)
apple.b2; // 可以訪問
apple.c2; // 可以訪問
apple.a3; // 報(bào)錯(cuò)
apple.b3; // 報(bào)錯(cuò)
apple.c3; // 報(bào)錯(cuò)
}
};
4 內(nèi)部類問題
如果一個(gè)類是另一個(gè)類的內(nèi)部類, 那么外部類的任何繼承關(guān)系都是對內(nèi)部類可見的, 也就不再受public, protected以及private的限制了
class Fruit{
class Apple{
//無論Fruit是public, protected還是private, 都在此處可見的
};
};