【Effective Modern C++】第三章 转向现代C++:12. 使用override重写声明函数

面向对象编程的基本理念是:在派生类中虚函数实现,会改写基类中对应虚函数的实现。

虚函数重写的约束条件

要想重写一个函数,必须满足下列要求:

  • 基类中的函数必须是虚函数。
  • 基类和派生类中的函数名字必须完全相同(析构函数例外)。
  • 基类和派生类中的函数形参类型必须完全相同。
  • 基类和派生类中的函数常量性必须完全相同。
  • 基类和派生类中的函数返回值和异常规格必须兼容。
  • 基类和派生类中的函数引用限定符必须完全相同。(C++11新增)

引用限定符:限定类的非静态成员函数只能被左值对象或右值对象调用。

&的成员函数仅允许左值对象调用;

&&的成员函数仅允许右值对象调用。

如果基类中的虚函数带有引用饰词,则派生类要对该函数进行改写版本必须也带有完全相同的引用饰词。否则相当于重新定义了一个新的函数。

编译器可能不报警但是存在错误的例子:

c++ 复制代码
class Base {
public:
    virtual void mf1() const;
    virtual void mf2(int x);
    virtual void mf3() &;
    void mf4() const;
};

class Derived: public Base {
public:
    virtual void mf1(); // mf1是声明为const的,而在派生类中没有
    virtual void mf2(unsigned int x); // mf2的形参类型是int,而在派生类中的则是unsigned int
    virtual void mf3() &&; // mf3带有左值引用限定符,而派生类中的则是右值引用限定符
    void mf4() const; // mf4未声明为虚函数
};

override 声明:显式保证重写正确性

通过在派生类函数后添加override声明,显式标明该函数意图改写基类虚函数版本:

  • 编译器会强制校验所有重写条件,不满足则直接编译报错(而非运行时才暴露问题);
  • 额外价值:修改基类虚函数签名时,可通过 "编译报错的派生类数量" 衡量影响面,判断修改代价是否值得。
    例:
c++ 复制代码
class Derived: public Base {
public:
    virtual void mf1() const override;  // 匹配const限定
    virtual void mf2(int x) override;   // 匹配int形参
    virtual void mf3() & override;      // 匹配左值引用限定
    void mf4() const override;          // 基类需补充virtual,否则仍报错
}; 

新增两个语境关键字:override和final

语言保留这两个关键字,但是仅在特定语境下保留。只在特定语境下用作关键字,其他情况下可用作变量名。

例:

c++ 复制代码
class Warning {         //C++98潜在的传统类代码
public:
    ...
    void override();    //C++98和C++11都合法(且含义相同)
    ...
};

class Base { 
public: 
	virtual void func() final; // 禁止派生类重写func 
}; 
class FinalClass final {}; // 禁止FinalClass被继承 
// class SubClass : public FinalClass {}; // 编译错误

关于成员函数引用限定符

编写一个函数仅接受传入左值实参,需要声明一个非const左值引用形参:

c++ 复制代码
void doSomething(Widget& w); // 仅接受左值的Widgets类型

编写一个函数仅接受传入右值实参,需要声明一个右值引用形参:

c++ 复制代码
void doSomething(Widegt& w); // 仅接受右值的Widgets类型

成员函数引用限定符的作用就是针对发起成员函数调用的对象,即*this进行区分(是左值对象还是右值对象);const限定符的作用就是约束*this的常量性(是否是const对象)

左值引用类型的重载版本返回的是一个左值引用(一个左值),右值引用类型的重载版本返回的是一个临时对象(一个右值)。

总结

  • 为需要重写的函数添加override声明。
  • 成员函数引用限定符使得对于左值和右值对象(*this)的处理能够区分开来。

原著在线阅读地址

相关推荐
端平入洛2 天前
delete又未完全delete
c++
端平入洛3 天前
auto有时不auto
c++
哇哈哈20213 天前
信号量和信号
linux·c++
多恩Stone3 天前
【C++入门扫盲1】C++ 与 Python:类型、编译器/解释器与 CPU 的关系
开发语言·c++·人工智能·python·算法·3d·aigc
蜡笔小马3 天前
21.Boost.Geometry disjoint、distance、envelope、equals、expand和for_each算法接口详解
c++·算法·boost
超级大福宝4 天前
N皇后问题:经典回溯算法的一些分析
数据结构·c++·算法·leetcode
weiabc4 天前
printf(“%lf“, ys) 和 cout << ys 输出的浮点数格式存在细微差异
数据结构·c++·算法
问好眼4 天前
《算法竞赛进阶指南》0x01 位运算-3.64位整数乘法
c++·算法·位运算·信息学奥赛
yyjtx4 天前
DHU上机打卡D31
开发语言·c++·算法
czxyvX4 天前
020-C++之unordered容器
数据结构·c++