公用,私有和受保护的继承
派生类可以定义0个或多个访问标号。
class classname:access-label class1[,access-label class2]
{
...
};
对类所继承的成员的访问由基类中的成员访问级别和派生类的**类派生列表**的访问标号共同控制。
每个类可以控制它所定义的成员的访问。一个类可以加强但不能放松对所继承的成员的访问。
- if public : 基类成员保持自己的访问级别
- 基类的public和protected成员在派生类中为protected成员
if private,基类的所有成员在派生类中为private成员
派生访问标号将控制派生类的用户对从
Base
继承而来的成员的访问。
接口继承与实现继承
public
是一种接口继承,而private
和protected
是一种实现继承。
什么时候使用继承
B 继承 A <==> B is an A
B 是 A 的一个成员 <==> A has a B.
private继承时去掉个别成员
如果在private
继承时希望使得某个继承而来的成员变量是public
的,那么可以用using baseclass::variable
从命名空间中使用名字。
struct默认为public继承,而class默认为private继承
继承与静态成员
如果基类定义了一个static
成员,那么在整个层次中只有一个这样的成员。无论从基类中派生出多少个派生类,每个static
成员只有一个实例。
且static
变量遵从常规访问控制。
引用转换不同于转换对象
将派生类对象传递给希望接受基类引用的函数时,没有创建新的对象,而是直接bind到该对象
将派生类对象传递给希望接受基类对象的函数时,会调用复制构造函数。
用派生类对象对基类对象初始化or赋值
对基类对象初始化or赋值时,实际上是在调用函数。
初始化时调用构造函数,赋值时调用赋值操作符。
从基类到派生类对象的转换是不存在的
当基类指针or引用实际bind到派生类对象时,从基类到派生类的转换也存在限制。(因为编译器在编译时无法确定转换是否合法,只会关注指针or引用的静态类型。)
基类构造函数和复制控制
继承对基类的唯一影响是:在确定提供哪些构造函数时,必须考虑一类新用户。像任意其它成员一样,如果基类希望提供只有派生类才能调用的构造函数,那么可以限制为protected
。
类是否需要定义复制构造函数完全取决于自己的需要
如:基类定义自己的复制构造函数而派生类使用编译器隐式合成的;派生类使用自己的定义的复制构造函数而基类使用隐式合成的。
如何定义派生类的复制构造函数
如果派生类显式定义自己的复制构造函数或赋值操作符,则该
定义将完全覆盖默认定义。被继承类的复制构造函数和赋值操
作符负责对基类成分以及类自己的成员进行复制或赋值。
同理,当派生类重载赋值操作符时,也需要显式初始化基类
派生类的析构函数
派生类的析构函数总是只释放派生类在基类之外定义的变量,再调用基类的析构函数
虚析构函数
如果析构函数为虚函数,那么通过指针调用时,运行哪个析构函数将因指针 所指对象类型的不同而不同。
构造函数和赋值操作符不能定义为虚函数
如果在构造函数或析构函数中调用虚函数,则运行的是为构造
函数或析构函数自身类型定义的版本。