菱形继承
两个派生类继承同一个基类
又有某个类同时继承两个派生类
这种继承被称为菱形继承,或者钻石继承
两个类有共同的父类,又有共同的子类
菱形继承问题:
- 羊继承了动物的数据,驼集成了动物的数据,当羊驼使用数据时,就会产生二义性。
- 羊驼继承自动物的数据继承了两份,但这个数据我们只需要一份就可以了。
利用虚继承可以解决菱形继承的问题。
在继承之前,加上关键字virtual,变为虚继承。
虚基类指针指向虚基类表,虚基类表中记录了偏移量,指向唯一的数据
菱形继承是C++中独有的
C++具备其它语言没有的多重继承的特性。
多重继承:一个子类继承多个父类,这些父类可能继承同一个父类。
多态
多态分为两类
- 静态多态:函数重载和运算符重载属于静态多态,复用函数名
- 动态多态:派生类和虚函数实现运行时多态
静态多态和动态多态的区别:
- 静态多态的函数地址早绑定——编译阶段确定函数地址
- 动态多态的函数地址晚绑定——运行阶段确定函数地址
动态多态满足条件:
- 有继承关系
- 子类要重写父类的虚函数(返回值、函数名、参数列表要完全相同)
动态多态的使用:
- 父类指针或引用指向子类对象:Animal &animal = cat;
当子类重写父类的虚函数时,子类中的虚函数表内部会替换成子类的虚函数地址。
当父类指针或引用指向子类对象的时候,发生多态
Animal & animal = cat;
开闭原则:对扩展进行开放,对修改进行关闭。
纯虚函数和抽象类
在多态中,通常父类中虚函数的实现都是毫无意义的,主要都是调用子类重写的内容。
因此可以将虚函数改为纯虚函数
纯虚函数写法:将{}改为= 0;
当类中有了纯虚函数,这个类也称为抽象类
抽象类特点:
- 无法实例化对象
- 子类必须重写抽象类中的纯虚函数,否则也属于抽象类
虚析构和纯虚析构
多态使用时,如果子类中有属性开辟到堆区,那么父类指针在释放时无法调用到子类的析构代码。
解决方法:将父类中的析构函数改为虚析构或纯虚析构
虚析构和纯虚析构共性:
- 可以解决父类指针释放子类对象
- 都需要有具体的函数实现
虚析构和纯虚析构区别:
- 如果是纯虚析构,该类属于抽象类,无法实例化对象
纯虚析构需要声明也需要实现