C++的访问权限问题

2019-01-10  本文已影响15人  Teci

一般在继承的时候, 子类是否能访问父类的成员变量, 需要分情况讨论

1. 同类不同模板

//下面这个例子跟继承没有关系, 不过先来探讨一下
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);    //此处调用a1对象的成员变量, 并传入一个A<int>对象, 没有问题
  a1.test(a2);    //此处调用a1对象的成员变量, 传入一个A<float>, 此时编译器将a1和a2认为是两个不同的类型, 此处报错为: 'data' is a private member of 'A<float>', 不过这个报错倒是因编译器而异
}

2. 形参类型与所在类的关系

2.1 形参类型与所在类类型相同

//在Apple的sale这个方法中, 传入的形参是Apple类型, 而sale方法本身也在Apple类中, 故此时形参类型与所在类类型相同
class Fruit {
private:
  int name;
protected: 
  int type;
};

class Apple : public Fruit {  //此处的public跟Apple中是否能访问Fruit的变量无关, 此处的public是标明从Fruit处继承来的变量的属性
public:
  void sale(Apple& apple){
      apple.name; //name是private的, 此处会报错, 无法访问
      apple.type; //type是protected的, 此处可以直接访问
  }
};

2.1 形参类型与所在类类型不同

//我们来看一个极端一点的例子
class Fruit{ 
protected:
    int name; 
};
class Apple : public Fruit {//虽然Apple继承于Fruit, 不过依然无法访问Fruit中protected的变量
public:
    void Method(Fruit& fruit) { //此处形参类型与类类型不同
       fruit.name;  //此处报错, 因为name是protected的
    }
};

3 访问控制符

3.1 当所在类类型与形参类型不同, 且没有继承关系时

//之前提到过在继承时的访问控制符起的功效是将父类中继承的变量, 按照访问控制符的类型加载到子类中
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是Apple类而当前所在类时Banana, 报错
    apple.b1; // 因为其是protected对象, 报错
    apple.c1; // 可以访问

    apple.a2; // 报错
    apple.b2; // 报错
    apple.c2; // 报错
    
    apple.a3; // 报错
    apple.b3; // 报错
    apple.c3; // 报错
  }
};

3.2 当所在类类型与形参类型不同, 但有继承关系时

//在每个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的继承关系时private的, ApplePear是无法获知该继承关系, 也就无法继承Fruit3中的值
public:
  void sale(Apple& apple){
    apple.a1; // 报错
    apple.b1; // 可以访问
    apple.c1; // 可以访问

    apple.a2; // 报错
    apple.b2; // 可以访问
    apple.c2; // 可以访问
    
    apple.a3; // 报错
    apple.b3; // 报错
    apple.c3; // 报错
  }
};

4 内部类问题

如果一个类是另一个类的内部类, 那么外部类的任何继承关系都是对内部类可见的, 也就不再受public, protected以及private的限制了

class Fruit{
  class Apple{
    //无论Fruit是public, protected还是private, 都在此处可见的
  };
};
上一篇 下一篇

猜你喜欢

热点阅读