静态多态:
早绑定 --->函数重载 在编译时就已经确定调用哪个函数
动态多态:
晚绑定 --->继承类间的同名函数调用,当父类指针指向子类对象时,在运行时才判定调用的是子类中的同名函数,而非父类。 加virtual 关键字,变为虚函数
【动态多态的内存泄露问题】
使用delete销毁父类指针,只会执行父类的析构函数
销毁子类指针,会执行父类和子类的析构函数
但在多态中,是用一个父类的指针指向一个子类的内存,因此无法执行子类的析构函数 Shape *s1=new Circle(3,4,5.0);
使用【虚析构函数】来解决
virtual不能修饰:
普通函数(非类内)
静态函数(static)
内联函数(inline)
构造函数
重载,隐藏,覆盖的区别:
(1)重载是指同一类中两个同名函数,但是参数的类型和数量不同(静态多态)
(2)隐藏是指子类对象调用函数时,对父类同名函数的隐藏。具有同名函数,即使参数不同。 如果要调用父类的该函数需要:子类对象.父类::函数
(3)覆盖是指父类指针指向子类对象的情况下,通过父类指针调用虚函数时,子类同名虚函数对父类同名虚函数的覆盖。具有同名的虚函数,且参数与返回类型也相同。(动态多态)
1.对象大小:
指类实例化对象时,数据成员占有内存空间的大小,不包括成员函数。没有任何数据成员的类理论上讲不占任何内存。
2.对象地址:
指通过一个类实例化一个对象时,对象占有一定的内存单元,这个对象占据的第一个单元的地址即是对象的地址
3.对象成员地址:
指当用一个类实例化一个对象时,这个对象可能有多个数据成员,由于数据成员的数据类型不同,那么占据的空间大小也就不同,这个类的每个成员都会占一定的空间,每个数据成员的地址也就是对象成员的地址不同
4.虚函数表指针:
指在有虚函数时,实例化一个对象时,这个对象的第一块内存中存放的是一个指针,是虚函数表的地址,大小为4。
5.如果实例化对象没有数据成员,就会有一个内存单元来标记这个对象的存在,如果对象中有数据成员,就不用内存单元来标记这个对象的存在,这个对象的大小就是数据成员所占内存的大小。
【抽象类】
含有纯虚函数的类,无法实例化对象,其子类也可以是抽象类,也可以不是(只有当子类不是抽象类,即对所有函数都实现了,才可以用这个子类实例化对象)
class Shape{
public:
virtual double calcArea(){return 0};//虚函数
virtual double calcPerimeter()=0;//纯虚函数(在后面加“=0”)
}
接口类:
仅含有纯虚函数的类称为接口类
(没有数据成员,只有成员函数,成员函数是纯虚函数)
RTTI 运行时类型识别(Run-Time Type Identification )
有的编译器需要手动包含头文件#include <typeinfo>
在父类指针指向子类对象时,为了能够调用子类对象的非虚函数,引入RTTI,进行对象类型识别与对象指针的转换