Item 34 分清对接口的继承和对实现的继承

本文探讨了C++中虚函数的不同类型及其在派生类中的作用。包括纯虚函数仅继承接口,简单虚函数同时继承接口与实现,以及非虚函数的接口和实现都被继承且不允许被覆盖的情况。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

需求:
1> 通过派生,只继承基类的函数接口(即它们的声明);
2> 接口和实现都继承,而且不让它们被派生类覆盖;
3> 接口和实现都继承,允许它们被派生类覆盖。

class Shape { // an abstract class public: virtual void draw() const = 0; // 纯虚函数:只继承接口 virtual void error(const std::string& msg); // 简单虚函数(有实现的虚函数;非纯虚函数):既继承接口,又给了一个默认的实现 int objectID() const; // 非虚函数:接口和实现都继承,必须和基类的行为一样 ... }; class Rectangle: public Shape { ... }; class Ellipse: public Shape { ... };

● 可以给Shape::draw()提供一个实现。调用的时候:

Shape *ps1 = new Rectangle; ps1->draw(); // calls Rectangle::draw Shape *ps2 = new Ellipse; ps2->draw(); // calls Ellipse::draw ps1->Shape::draw(); // calls Shape::draw ps2->Shape::draw(); // calls Shape::draw

● 简单虚函数比较危险。
如果客户忘记重定义该函数,该函数也能工作,没有人能提醒客户。只有到工作不正常的时候,客户才能知道。故而不提倡使用这种声明方式。或者改进一下,把它变成纯虚函数,并提供一个默认实现:

class Airplane { public: virtual void fly(const Airport& destination) = 0; ... protected: void defaultFly(const Airport& destination); }; void Airplane::defaultFly(const Airport& destination) {...}

这样,编译器会提醒客户。而且,defaultFly应该是一个“非虚函数”。这一点很重要:不能让客户重定义它!否则它一样有可能会被客户忘记。

不过,又有fly又有defaultFly,这样相似的名字会影响客户的理解。随着函数的增多,大量的defaultXXX会污染类的名字空间。解决方案:

class Airplane { public: virtual void fly(const Airport& destination) = 0; // 基类里还是定义它为“纯虚函数” ... }; void Airplane::fly(const Airport& destination) {...} // 给这个纯虚函数一个实现,该实现作为派生类的默认实现 class ModelA: public Airplane { public: virtual void fly(const Airport& destination) { Airplane::fly(destination); } ... }; class ModelB: public Airplane { public: virtual void fly(const Airport& destination) { Airplane::fly(destination); } ... }; class ModelC: public Airplane { public: virtual void fly(const Airport& destination); ... }; void ModelC::fly(const Airport& destination) { // 派生类的自定义行为 }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值