目录
多态分为静态多态和动态多态
- 静态多态:函数重载和运算符重载,复用函数名
- 动态多态:派生类和虚函数实现运行时动态多态
区别:
- 静态多态的函数地址早绑定,在编译阶段确定函数地址
- 动态多态的函数地址晚绑定,在运行阶段确定函数地址
语法
在speak()函数前加virtual变成虚函数
此时运行结果就能变成Cat Speak
动态多态满足条件
- 存在继承关系
- 子类重写父类的虚函数(不同于函数重载,函数重载是函数名相同参数不同,而重写的函数返回值类型和函数名和参数完全一致)
函数多态使用
父类的指针或引用,指向子类对象
多态的原理剖析
加virtual之前,Animal类占1个字节
加virtual之后,Animal类占4个字节
这4个字节是一个指针
例:虚函数实现计算器类
//实现计算机的抽象类
class AbstructCalculator {
public:
int num1; //操作数1
int num2; //操作数2
virtual int getResult() {
return 0;
}
};
//加法计算器类
class AddCalculator:public AbstructCalculator{
int getResult() {
return num1+num2;
}
};
//减法计算器类
class SubCalculator :public AbstructCalculator {
int getResult() {
return num1 - num2;
}
};
int main() {
AbstructCalculator* abc = new AddCalculator;
abc->num1 = 10;
abc->num2 = 10;
cout<<abc->getResult();
delete abc;
}
纯虚函数和抽象类
在多态中,通常父类中虚函数的实现是毫无意义的,主要都是调用子类重写的内容
因此可以将虚函数改为纯虚函数
语法:
virtual 返回值类型 函数名(参数列表)=0;
当类中有了纯虚函数,这个类也称为抽象类
特点:
- 无法实例化对象
- 子类必须重写抽象类中的纯虚函数,否则也属于抽象类
虚析构和纯虚析构
多态使用时,如果子类中有属性开辟到堆区,那么父类指针在释放时无法调用到子类的析构代码
因此,将父类中的析构函数改为虚析构或纯虚析构(这样就变成抽象类)
- 可以解决父类指针释放子对象
- 需要有具体的函数实现
类名::~类名(){}