文章目录
确定你的public继承塑模出is-a关系
避免遮掩继承而来的名称
区分接口继承和实现继承
考虑virtual函数以外的其他选择
绝不重新定义继承而来的non-virtual函数
绝不重新定义继承而来的缺省参数值
通过复合塑模出has-a或“根据某物实现出”
明智而审慎地使用private继承
明智而审慎地使用多重继承
Is A
public继承意味着 is a
。适用于base classes身上的每一件事情一定也适用于derived classes身上,因为每一个derived class对象也都是一个base class对象:
class Bird
{
public:
virtual void fly() = 0;
virtual void fly(int time);
void jump();
void jump(int time);
...
private:
int x;
};
// Penguin is a Bird
class Penguin: public Bird
{
public:
virtual void fly();
void jump();
...
};
Penguin重写的方法会遮掩基类Bird方法,即使函数参数不同:
Bird作用域 | Penguin作用域 | |
---|---|---|
fly | fly()、fly(int) | fly() |
jump | jump()、jump(int) | jump() |
Penguin p;
int x;
...
p.fly(); // success
p.fly(x); // error
p.jump(); // success
p.jump(x); // error
为解决这个问题可以引入 using
说明:
class Penguin: public Bird
{
public:
using Bird::fly;
using Bird::jump; // 让Bird内名为fly和jump的所有东西在Penguin内都可见
virtual void fly();
void jump();
...
};
// 调用
Penguin p;
int x;
...
p.fly(); // success
p.fly(x); // success
p.jump(); // success
p.jump(x); // success
此章内容在《大话数据结构》中体现得更好,所以多数内容暂时略去。后续将针对《大话数据结构》作相关笔记。