转向现代C++——在意为改写的函数添加 override

文章目录

    • [在意为改写的函数添加 override](#在意为改写的函数添加 override)
      • [未添加 `override`](#未添加 override)
      • [显式添加 `override`](#显式添加 override)
      • 引用限定符

在意为改写的函数添加 override

虚函数改写(Overriding)是实现多态的核心,但它的隐式匹配规则非常严苛。如果开发者稍有疏忽,本意是"改写(Override) "的函数就会变成"重载(Overload) "或"隐藏(Hide)",而编译器对此往往保持沉默。
虚函数改写的严格条件

在 C++ 中,派生类函数要想真正改写基类的虚函数,必须满足以下所有条件:

  1. 函数名相同
  2. 形参类型完全相同
  3. 常量性( const属性)完全相同
  4. 返回值类型和异常规格(Exception specification)必须兼容
  5. (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?

  1. 让编译器做打字工 只要有任何一处不匹配(比如漏了 const,改了参数类型),编译器会立刻拒绝编译 ,并给出精准的错误提示。
  2. 无成本的代码文档 任何阅读代码的人一眼就能看出哪些函数是多态的关键节点,无需频繁跳回基类去确认。
  3. 安全重构基类 如果哪天你需要修改基类 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;  // 正确:成功改写左值版本
};
相关推荐
王老师青少年编程1 小时前
csp信奥赛C++高频考点专项训练之前缀和&差分 --【一维差分】:[NOIP 2018 提高组] 铺设道路
c++·前缀和·差分·csp·高频考点·信奥赛·铺设道路
星马梦缘1 小时前
aaaaa
数据结构·c++·算法
喵星人工作室2 小时前
C++火影忍者1.1.2
开发语言·c++
basketball6162 小时前
C++ 中的 ptrdiff_t 详解
开发语言·c++
wunaiqiezixin3 小时前
互斥锁与自旋锁的区别
c++
月亮邮递员6163 小时前
Markdown语法总结
开发语言·前端·javascript
printfLILEI3 小时前
php中的类与对象以及反序列化
linux·开发语言·php
曹牧3 小时前
C#:主线程能够捕获到子线程中的异常
开发语言·数据库·c#
代码中介商3 小时前
深入解析STL中的stack、queue与priority_queue
开发语言·c++