C++ primer第四版15.3小节有这样一段话:
如果是 public 继承,则用户代码和后代类都可以使用派生类到基类的转换。如果类是使用 private 或 protected 继承派生的,则用户代码不能将派生类型对象转换为基类对象。如果是 private 继承,则从 private 继承类派生的类不能转换为基类。如果是 protected 继承,则后续派生类的成员可以转换为基类类型。
看完后感觉一团雾水,后来看英文原版中这样写:
If the inheritance is public, then both user code and member functions of subsequently derived classes may use the derived-to-base conversion. If a class is derived using private or protected inheritance, then user code may not convert an object of derived
type to a base type object. If the inheritance is private, then classes derived from the privately inherited class may not convert to the base class. If the inheritance is protected, then the members of subsequently derived classes may convert to the base
type.
这样写我们就清楚多了。为了更明白的说明问题,我们构造三个类A,B,C。A是base类。B继承于A,C继承于B。
现在我们要说明的问题有两个,一个是用户代码中访问的问题,一个是派生类中访问的问题。
(1)先说用户代码中访问的问题。
B和A的关系有三种:
1 B:public A
2 B:proteced A
3 B:private A
针对这样一段代码:
A a;
B b;
a = b;
我们把它放在main.cpp中时,只有1是正确的,2,3都是错误的。这个说明的就是:只有当子类和基类的关系是public继承时,才能在用户代码中把子类赋给基类,其余两种都不行。
(2) 后继派生类中访问的问题
C和B的关系有三种
1 C:public B
2 C:proteced B
3 C:private B
但是不管C和B关系怎样,C中总是可以访问B的public和protected成员,现在假设C中有一个成员函数
void C::func()
{
A a;
B b;
a = b;
}
此时当B和A的关系是public,protected继承的时候,函数func的这段代码都没有问题,因为A,B中的非private成员在C中都可以为访问,所以没有问题。但是当B和A的关系是private的时候,这段代码就出问题了,因为A在B中是私有成员,所以不能访问A,更不用说 a = b了。
相关文章
- 10-24不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成员)
- 10-24c++ 派生类向基类转换的可访问性
- 10-24C++ 派生类到基类转换的可访问性
- 10-24C++:调整基类成员在派生类中的访问属性的其他方法(同名成员和访问声明)
- 10-243.3 C++改变基类成员在派生类中的访问属性
- 10-24C++ - 派生类访问模板化基类(templatized base class)的命名
- 10-24C++中派生类对基类成员的访问形式
- 10-24C++ 派生类到基类转换的可访问性
- 10-24C++中派生类对基类成员的访问形式