文章目录
-
- [在意为改写的函数添加 override](#在意为改写的函数添加 override)
-
- [未添加 `override`](#未添加
override) - [显式添加 `override`](#显式添加
override) - 引用限定符
- [未添加 `override`](#未添加
在意为改写的函数添加 override
虚函数改写(Overriding)是实现多态的核心,但它的隐式匹配规则非常严苛。如果开发者稍有疏忽,本意是"改写(Override) "的函数就会变成"重载(Overload) "或"隐藏(Hide)",而编译器对此往往保持沉默。
虚函数改写的严格条件
在 C++ 中,派生类函数要想真正改写基类的虚函数,必须满足以下所有条件:
- 函数名相同 。
- 形参类型完全相同 。
- 常量性( const属性)完全相同 。
- 返回值类型和异常规格(Exception specification)必须兼容 。
- (C++11 新增) 引用限定符(Reference qualifiers)完全相同 。
未添加 override
cpp
class Base {
public:
virtual void doWork(); // 基础虚函数
virtual void process(int x); // 带参数
virtual void display() const; // const 函数
};
class Derived : public Base {
public:
// 错误 1:漏掉了 const,变成了一个全新的虚函数,隐藏了 Base::doWork
virtual void doWork();
// 错误 2:参数类型写错(int 变成了 unsigned int),变成了重载(Overload)
virtual void process(unsigned int x);
// 错误 3:函数名大小写写错,变成了完全无关的新函数
virtual void DisPlay() const;
};
:::color4
- 编译期无警告****: 上述
Derived中的三个函数,在 C++98/C++11 语法下完全合法。编译器会认为你是故意在派生类里增加新函数的。 - 运行时行为诡异(多态失效)
:::
cpp
Base* bp = new Derived();
bp->process(10); // 期望调用 Derived::process,但实际上调用了 Base::process!
显式添加 override
通过在派生类函数后面加上 override 关键字,向编译器明确传达了意图:"我这个函数绝对是改写基类的,请帮我检查!"
cpp
class DerivedCorrected : public Base {
public:
// 编译期报错!Base 中没有非 const 的 doWork()
void doWork() override;
// 编译期报错!Base 中没有接收 unsigned int 的 process
void process(unsigned int x) override;
// 编译期报错!Base 中没有名为 DisPlay 的函数
void DisPlay() const override;
// ---------------- 正确的改写 ----------------
void display() const override; // 完美匹配!编译通过
};
为什么强制使用 override?
- 让编译器做打字工 : 只要有任何一处不匹配(比如漏了 const,改了参数类型),编译器会立刻拒绝编译 ,并给出精准的错误提示。
- 无成本的代码文档 : 任何阅读代码的人一眼就能看出哪些函数是多态的关键节点,无需频繁跳回基类去确认。
- 安全重构基类 : 如果哪天你需要修改基类 Base::doWork() 的签名(比如加个参数),所有未同步修改的派生类都会在编译期报警,而不会在运行时悄悄崩溃。
引用限定符
C++11 允许根据对象是左值还是右值来调用不同的成员函数。如果基类指定了引用限定符,派生类改写时也必须完全一致,否则也会变成全新的函数。
cpp
class Widget {
public:
// 只有左值对象 (*this 是左值) 才能调用此函数
virtual void data() &;
// 只有右值对象 (*this 是右值) 才能调用此函数
virtual void data() &&;
};
class SubWidget : public Widget {
public:
// 如果不加 override,这只是一个没有引用限定的新函数,会隐藏 Widget::data
// 加上 override 后,编译器会立刻发现你漏掉了 & 或 && 从而报错
void data() override; // 错误:无法通过编译
void data() & override; // 正确:成功改写左值版本
};