多态
- 概括:一个函数可以有多种形态
- 分类:
- 静态多态:函数重载、运算符重载都是静态多态,函数地址早绑定,编译阶段确定地址
- 动态多态:函数地址晚绑定,在运行阶段确定地址
动态多态
- 实现条件:
- 需要有继承关系存在,以及虚函数的存在,virtual修饰的成员函数叫做虚函数
- 子类需要重写父类的虚函数。重写:函数的返回值、名字、参数都要一致
- 需要使用父类指针指向子类对象 或者 使用父类引用子类对象
class base
{
virtual void fun(){}
};
class son:public base
{
virtual void fun(){}
};
son s1;
base *b1=&s1;
base &b2=s1;
- C++推荐使用多态设计方法
- 代码可读性强、逻辑性强
- 代码结构清晰
- 利于代码后期的扩展与维护,对代码扩展开发,对代码修改屏蔽
- C++允许子类转父类,不允许父类转子类
纯虚函数
- 作用:在实现多态过程中,发现父类的虚函数基本不会被调用,实现函数没有多大意义,因此可以写成纯虚函数。纯虚函数不用具体实现。
- 语法:virtual void fun()=0;
class base
{
virtual void fun()=0;//纯虚函数
int n1;
int n2;
};
class son:public base
{
void fun(){}
};
base b1;//不能实例化
son s1;//可以实例化对象
抽象类
- 只要有纯虚函数的类就是抽象类,抽象类不能实例化对象
- 如果子类没有重写父类的纯虚函数,子类也是抽象类,不能实例化对象。所以,所有纯虚函数都需要重写。
虚析构与纯虚析构
- 作用:在父类指针指向堆区子类对象时,但最后delete释放父类指针指向的空间时,子类的析构不会被调用。因此可以使用虚析构或纯虚析构方式让子类空间得到释放
base *b1=new son();
delete b1;
b1=nullptr;
- 实现方式
虚析构:在析构函数前加一个virtual
virtual ~base()
{
};
纯虚析构:在析构函数前加一个virtual,最后添加=0; 但是纯虚析构需要在类外实现,有纯虚析构的类不能实例化对象,因为是一个抽象类。
virtual ~base()=0;
base::~base()
{
cout<<""<<endl;
}