15-4 抽象基类
类的继承关系图:
抽象基类
含有纯虚函数的类成为抽象基类,用来表示一个通用概念
如此处Disc_count表示折扣书籍的通用概念,它的虚函数net_price为纯虚函数,表示折扣方案,它的派生类如Bulk_quote表示每个具体的折扣方案
- 纯虚函数只需声明,不必实现,末尾加上
=0
- 抽象基类不能实例化,即无法建立抽象基类的对象
- 抽象基类的子类必须实现纯虚函数,否则仍是抽象基类
//用于保存折扣值和购买量的类,派生类使用这些数据可以实现不同的价格策略
class Disc_quote : public Quote {
public :
Disc_quote () = default;
Disc_quote(const string &book, double price,
int qty, double disc) :
Quote(book, price),
quantity(qty), discount(disc) {}
// = 0 : 声明为纯虚函数
//由于基类Quote中net_price已经声明为虚函数
//所以此处不必再加关键字virtual
double net_price(int) const = 0;
protected :
int quantity = 0; //折扣适用的购买量
double discount = 0.0; //表示折扣的小数值
};
Disc_quote discounted; //错误:不能定义抽象基类的对象
Bulk_quote bulk; //正确:Bulk_quote没有纯虚函数
派生类构造函数只初始化它的直接基类
定义Bulk_quote类,它继承自Disc_quote类
class Bulk_quote : public Disc_quote {
public :
Bulk_quote() = default;
Bulk_quote(const string &book, double price,
int qty, double disc) :
Disc_quote(book, price, qty, disc);
//覆盖抽象基类中的纯虚函数,实现一种具体的折扣策略
double net_price(int ) const override;
};
Bulk_quote的对象构成以及初始化流程
初始化流程类似于递归:
每个类各自控制其对象的初始化过程。
- Bulk_quote构造函数传递实参,调用Disc_quote的构造函数
- 随后 Disc_quote 的构造函数继续调用Quote的构造函数。
- Quote 的构造函数首先初始化bulk的bookNo和price成员【Bulk_quote中的Quote子对象的成员】
- 当Quote 的构造函数结束后,开始运行Disc_quote 的构造函数并初始化 quantity和discount成员【Bulk_quote中的Disc_quote子对象的成员】
- 最后运行Bulk _quote 的构造函数,该函数无须执行实际的初始化或其他工作。【Bulk_quote无新增成员】