当前位置: 首页 > article >正文

C++11 中 final 和 override 从入门到精通

文章目录

    • 一、引言
    • 二、`final` 关键字
      • 2.1 `final` 关键字的基本概念
      • 2.2 `final` 关键字的语法
      • 2.3 `final` 关键字的使用示例
        • 2.3.1 防止类被继承
        • 2.3.2 防止虚函数被重写
      • 2.4 `final` 关键字的使用场景
      • 2.5 `final` 关键字的注意事项
    • 三、`override` 关键字
      • 3.1 `override` 关键字的基本概念
      • 3.2 `override` 关键字的语法
      • 3.3 `override` 关键字的使用示例
      • 3.4 `override` 关键字的作用
      • 3.5 `override` 关键字的使用场景
      • 3.6 `override` 关键字的注意事项
    • 四、`final` 和 `override` 的组合使用
    • 五、总结

一、引言

在 C++ 编程的世界里,继承和多态是面向对象编程的核心特性。通过继承,我们可以创建新的类,复用现有类的代码;通过多态,我们可以以统一的方式处理不同类型的对象。然而,在实际开发中,我们可能会遇到一些问题,比如不小心重写了基类的虚函数,或者希望某个类不能被继承,某个虚函数不能被重写。为了解决这些问题,C++11 引入了两个重要的关键字:finaloverride。这两个关键字的引入,使得我们在处理类的继承和虚函数的重写时更加灵活和安全。

二、final 关键字

2.1 final 关键字的基本概念

final 关键字用于限制类的继承和虚函数的重写。当 final 用于类时,表示该类不能被继承;当 final 用于虚函数时,表示该虚函数不能在派生类中被重写。

2.2 final 关键字的语法

  • 修饰类
class ClassName final {// 类的成员
};
  • 修饰虚函数
class Base {
public:virtual void func() final;
};

2.3 final 关键字的使用示例

2.3.1 防止类被继承
class Base final {
public:void foo() {std::cout << "Base::foo()" << std::endl;}
};// 下面的代码将无法通过编译,因为 Base 被声明为 final
// class Derived : public Base {
// public:
//     void foo() {
//         std::cout << "Derived::foo()" << std::endl;
//     }
// };

在这个例子中,Base 类被声明为 final,因此不能被其他类继承。如果尝试继承 Base 类,编译器会报错。

2.3.2 防止虚函数被重写
class Base {
public:virtual void foo() {std::cout << "Base::foo()" << std::endl;}virtual void bar() final {std::cout << "Base::bar()" << std::endl;}
};class Derived : public Base {
public:void foo() override {std::cout << "Derived::foo()" << std::endl;}// 下面的代码将无法通过编译,因为 bar() 在 Base 中被声明为 final// void bar() override {//     std::cout << "Derived::bar()" << std::endl;// }
};

在这个例子中,Base 类的 bar() 函数被声明为 final,因此在 Derived 类中不能重写该函数。如果尝试重写 bar() 函数,编译器会报错。

2.4 final 关键字的使用场景

  • 设计意图明确:明确表示类或函数不应被进一步扩展或修改。
  • 防止意外重写:避免派生类意外重写基类的重要虚函数。
  • 优化机会:编译器可以对标记为 final 的虚函数进行去虚拟化优化。
  • 接口控制:在框架或库设计中控制哪些部分可以被用户扩展。

2.5 final 关键字的注意事项

  • final 不是虚函数声明的一部分,可以放在函数声明的任何位置(在参数列表后或函数体前)。
  • final 只能用于虚函数和类。
  • final 是一个标识符,在 C++11 之前可以用作变量名等,但在 C++11 及以后它有特殊含义。

final关键字在C++中的使用

三、override 关键字

3.1 override 关键字的基本概念

override 关键字用于显式地表明派生类中的成员函数是重写基类中的虚函数。使用 override 关键字可以提高代码的可读性和安全性,帮助开发者避免一些常见的错误。

3.2 override 关键字的语法

class Base {
public:virtual void func();
};class Derived : public Base {
public:void func() override;
};

3.3 override 关键字的使用示例

class Base {
public:virtual void display() const {std::cout << "Base class display" << std::endl;}
};class Derived : public Base {
public:void display() const override {std::cout << "Derived class display" << std::endl;}
};int main() {Base* basePtr = new Derived();basePtr->display(); // 输出: Derived class displaydelete basePtr;return 0;
}

在这个例子中,Derived 类的 display() 函数重写了 Base 类的 display() 函数,并且使用了 override 关键字。这向编译器明确表明,display() 是对基类虚函数的重写。

3.4 override 关键字的作用

  • 防止函数签名不匹配:如果函数签名不完全匹配(比如参数类型或个数不同),编译器会发出错误提示。
  • 防止函数名拼写错误:如果函数名拼写错误或参数列表有误,编译器也会给出错误提示。

3.5 override 关键字的使用场景

当派生类需要重写基类中的虚函数时,通常会使用 override 关键字。这种用法可以提高代码的可读性和安全性,防止因为函数签名不同而导致的错误。

3.6 override 关键字的注意事项

  • override 只能用于派生类中重写基类的虚函数。
  • 如果基类中没有对应的虚函数,或者函数签名不匹配,编译器将报错。

override关键字在C++中的使用

四、finaloverride 的组合使用

finaloverride 可以组合使用,表示一个虚函数在派生类中被重写,并且在该派生类中是最终的,不能被进一步重写。

class Base {
public:virtual void func() {std::cout << "Base::func()" << std::endl;}
};class Derived : public Base {
public:void func() override final {std::cout << "Derived::func()" << std::endl;}
};class MoreDerived : public Derived {
public:// 下面的代码将无法通过编译,因为 func() 在 Derived 中被声明为 final// void func() override {//     std::cout << "MoreDerived::func()" << std::endl;// }
};

在这个例子中,Derived 类的 func() 函数重写了 Base 类的 func() 函数,并且使用了 override 关键字。同时,func() 函数被声明为 final,因此在 MoreDerived 类中不能重写该函数。

五、总结

finaloverride 是 C++11 引入的两个非常有用的关键字,它们为类的继承和多态机制提供了更多的控制权和明确性。final 关键字用于指示一个类或成员函数不能被继承或覆盖,而 override 关键字用于明确指出派生类中的成员函数旨在覆盖基类中的同名虚拟函数。

通过对 finaloverride 关键字的理解和运用,我们可以更加安全、清晰地设计和实现 C++ 程序。在实际开发中,合理地使用这两个关键字可以提高代码的可读性、安全性和可维护性,避免一些潜在的错误。希望本文能帮助你更好地掌握 finaloverride 关键字的使用,让你的 C++ 编程之路更加顺畅。

http://www.lryc.cn/news/2401970.html

相关文章:

  • 跨多个微服务使用 Redis 共享数据时,如何管理数据一致性?
  • Linux网络——socket网络通信udp
  • 大数据-275 Spark MLib - 基础介绍 机器学习算法 集成学习 随机森林 Bagging Boosting
  • 大模型微调技术全景图:从全量更新到参数高效适配
  • c++ chrono头文件含义
  • git互联GitHub 使用教程
  • Python爬虫与Java爬虫深度对比:从原理到实战案例解析
  • 汇编语言综合程序设计:子程序、分支与循环深度解析
  • SpringBoot+Mysql实现的停车场收费小程序系统+文档
  • 面向对象进阶 | 深入探究 Java 静态成员与继承体系
  • 人脸识别技术成为时代需求,视频智能分析网关视频监控系统中AI算法的应用
  • 青岛国瑞数据采集网关软件平台:工业互联的智能基石——安全、高效、开放,驱动企业数字化转型
  • Git的由来与应用详解:从Linux内核到现代开发的革命性工具
  • @Prometheus 监控-MySQL (Mysqld Exporter)
  • pc端小卡片功能-原生JavaScript金融信息与节日日历
  • 窗口聚合窗口聚合
  • es在Linux安装
  • Go语言学习-->第一个go程序--hello world!
  • 高雄市12岁以下身心障碍儿童口腔保健合作院所名单数据集
  • Spring Boot 自动参数校验
  • 破局新能源消纳难题!安科瑞智慧能源平台助力10KV配电网重构未来
  • 推荐10个AI视频生成工具网站
  • TIA博途中的程序导出为PDF格式的具体方法示例
  • 【大模型:知识图谱】--4.neo4j数据库管理(cypher语法1)
  • Java 实现下拉框树状结构接口的核心思路
  • 数字化时代养老机构运营实训室建设方案:养老机构运营沙盘实训模块设计
  • 自由开发者计划 004:创建一个苹果手机长截屏小程序
  • 【Go语言基础】基本语法
  • 工作流引擎-18-开源审批流项目之 plumdo-work 工作流,表单,报表结合的多模块系统
  • 【虚拟机版本号】如果忘记了版本号,这样查找版本号