多态
如果子类中定义了与父类中原型相同的函数会发生什么?
函数重写
在子类中定义与父类中原型相同的函数
函数重写只发生在父类与子类之间
重载与重写区别:
重载:同一个作用域;
子类无法重载父类函数,父类同名函数将被覆盖;
重载是在编译期间根据参数类型和个数决定;
重写:发生于父类、子类之间;
父类和子类函数有相同的函数原型;
使用virtual关键字声明后能够产生多态;
运行期间根据具体对象类型决定调用的函数。
C++中通过virtual关键字对多态进行支持
使用virtual声明的函数被重写后即可展现多态特性
多态成立三个条件:
要有继承
要有虚函数重写
用父类指针指向子类对象
、联编是指一个程序模块、代码之间互相关联的过程。
2、静态联编(static binding),是程序的匹配、连接在编译阶段实现,也称为早期匹配。
重载函数使用静态联编。
3、动态联编是指程序联编推迟到运行时进行,所以又称为晚期联编(迟绑定)。
switch 语句和 if 语句是动态联编的例子。
在什么情况下应当声明虚函数
构造函数不能是虚函数。建立一个派生类对象时,必须从类层次的根开始,沿着继承路径逐个调用基类的构造函数
析构函数可以是虚的。虚析构函数用于指引 delete 运算符正确析构动态对象
虚析构函数:通过父类指针释放子类对象
说明1:
通过虚函数表指针VPTR调用重写函数是在程序运行时进行的,因此需要通过寻址操作才能确定真正应该调用的函数。而普通成员函数是在编译时就确定了调用的函数。在效率上,虚函数的效率要低很多。
说明2:
出于效率考虑,没有必要将所有成员函数都声明为虚函数
说明3 :
C++编译器,执行run函数,不需要区分是子类对象还是父类对象
构造的顺序是先构造父类、再构造子类
当调用父类的构造函数的时候,虚函数指针vfptr 指向父类的虚函数表
当父类构造完,调用子类的构造函数的时候,虚函数指针 vfptr 指向子类的虚函数表
结论:构造函数中无法实现多态
指针也是一种数据类型,C++类对象的指针p++/--,仍然可用。
指针运算是按照指针所指的类型进行的。
父类p++与子类p++步长不同;不要混搭,不要用父类指针++方式操作子类对象数组
纯虚函数为派生类提供一个公共接口
Virtual 类型 函数名(参数表) = 0;
一个具有纯虚函数的基类叫做抽象类
工程上的多继承
被实际开发经验抛弃的多继承
工程开发中真正意义上的多继承是几乎不被使用的
多重继承带来的代码复杂性远多于其带来的便利
多重继承对代码维护性上的影响是灾难性的
在设计方法上,任何多继承都可以用单继承代替