详细全面讲解C++中重载、隐藏、覆盖的区别

文章目录

      • 总结
      • 1、重载
        • 示例代码
        • 特点
        • 1. 模板函数和非模板函数重载
        • 2. 重载示例与调用规则
        • 示例代码
        • 调用规则解释
        • 3. 特殊情况与注意事项
            • 二义性问题
          • 函数特化与重载的交互
      • 2. 函数隐藏(Function Hiding)
        • 概念
        • 示例代码
        • 特点
      • 3. 函数覆盖(重写,Function Overriding)
        • 概念
        • 示例代码
        • 特点

总结

1、重载:在同一个作用域,函数名相同,参数列表不同,与返回值无关
2、隐藏:在基类和派生类之间发生的关系,函数名相同,派生类的函数把基类的函数给隐藏了,只关注函数数名,不管返回值和参数
3、覆盖:覆盖是隐藏的一种特殊情况,派生类和基类的函数,1 函数名相同;2 返回值相同;3 参数列表相同(不包括this指针在内);4 基类函数为虚函数;即为覆盖。

1、重载

  • 函数重载是指在同一个作用域(通常是在同一个类中,也可以是在全局作用域下)内,存在多个同名函数,但它们的参数列表(参数的个数、类型或者顺序)不同。编译器会根据调用时传递的实际参数来决定调用哪个具体的重载函数,函数重载实现了用同一个函数名来执行相似但参数有所不同的操作,提高了代码的可读性和易用性。
    总结来说就是:在同一个作用域,函数名相同,参数列表不同,与返回值无关
示例代码
#include <iostream>

// 重载函数,参数个数不同
int add(int num1, int num2) {
    return num1 + num2;
}

int add(int num1, int num2, int num3) {
    return num1 + num2 + num3;
}

// 重载函数,参数类型不同
double add(double num1, double num2) {
    return num1 + num2;
}

int main() {
    std::cout << add(2, 3) << std::endl;
    std::cout << add(2, 3, 4) << std::endl;
    std::cout << add(2.5, 3.5) << std::endl;
    return 0;
}

在上述代码中,add 函数有多个重载形式,有的重载函数参数个数不同(如两个 int 参数和三个 int 参数的版本),有的重载函数参数类型不同(如 int 参数和 double 参数的版本)。在 main 函数中调用 add 函数时,编译器根据传入的实际参数情况来选择匹配的重载函数进行调用。

特点
  • 作用域相同:重载的函数必须在同一个作用域内定义,比如都在某个类的内部或者都在全局作用域中。
  • 函数名相同:这是重载的关键特征之一,多个函数共享同一个函数名,方便代码的调用和理解,从使用者角度看好像是同一个函数根据不同情况执行不同逻辑。
  • 参数列表有差异:参数个数、类型或者顺序至少有一项不同,而返回类型不同不能作为函数重载的依据(因为仅返回类型不同时,编译器无法仅根据调用情况准确判断该调用哪个函数)。

除了以上常规的重载,还有一些同学认为泛型编程也存在重载,即模板函数和非模板函数的重载,
在C++中,模板函数与非模板函数之间可以存在重载关系,以下是关于它们重载的详细介绍:

1. 模板函数和非模板函数重载
  • 重载机制:重载允许在同一作用域内存在多个同名函数(对于函数重载而言,这些函数的参数列表有所不同,而返回类型不能作为区分重载的唯一依据),编译器会根据调用时实际传入的参数情况来决定调用哪一个具体的函数。模板函数和非模板函数的重载就是利用了这一机制,在合适的场景下,编译器会依据调用参数去选择调用模板函数版本还是非模板函数版本。
2. 重载示例与调用规则
示例代码
#include <iostream>

// 非模板函数
int add(int num1, int num2)
 {
    return num1 + num2;
}

// 模板函数
template<typename T>
T add(T num1, T num2)
 {
    return num1 + num2;
}

int main()
 {
    int result1 = add(3, 5);  // 调用非模板函数 add(int, int)
    double result2 = add(3.5, 2.5);  // 调用模板函数 add<double>(double, double)
    std::cout << "整数相加结果: " << result1 << std::endl;
    std::cout << "浮点数相加结果: " << result2 << std::endl;
    return 0;
}
调用规则解释
  • 精确匹配优先:当进行函数调用时,编译器首先会寻找参数类型与函数参数列表能精确匹配的非模板函数。在上述代码的 add(3, 5) 调用中,非模板函数 add(int, int) 的参数类型刚好和传入的两个整数参数完全匹配,所以编译器优先选择调用这个非模板函数,而不会去考虑模板函数版本,即使模板函数也能够通过实例化(将 T 实例化为 int)来处理这两个整数参数。
  • 模板函数实例化匹配:如果没有找到精确匹配的非模板函数,编译器会尝试对模板函数进行实例化,看能否通过实例化后的模板函数来匹配调用参数。例如在 add(3.5, 2.5) 调用中,不存在参数类型为两个 double 的非模板函数 add,此时编译器会查看模板函数,将模板参数 T 实例化为 double,生成 add(double, double) 这样一个实例化后的函数版本,它能很好地匹配传入的两个 double 类型参数,所以就调用这个实例化后的模板函数版本。
3. 特殊情况与注意事项
二义性问题
  • 当模板函数和非模板函数的参数匹配存在模糊情况时,可能会导致编译错误,出现二义性问题。例如:
#include <iostream>

// 非模板函数
void func(int num)
 {
    std::cout << "非模板函数func(int)" << std::endl;
}

// 模板函数
template<typename T>
void func(T num) 
{
    std::cout << "模板函数func(T)" << std::endl;
}

int main() 
{
    func(5);  // 编译错误,存在二义性,不知道该调用模板函数还是非模板函数
    return 0;
}

在这个例子中,调用 func(5) 时,传入的整数 5 既可以匹配非模板函数 func(int),也可以通过将模板函数的 T 实例化为 int 来匹配模板函数 func(T),编译器无法确定到底该调用哪一个函数,就会报二义性的编译错误。要解决这类问题,可以通过显式指定模板参数(如 func<int>(5) 就会明确调用模板函数版本)或者调整函数的参数类型等方式,使得调用具有明确的匹配对象。这个问题并不是所有的编译器都存在,在VS2022就不存在该问题。

函数特化与重载的交互
  • 函数模板可以进行特化,即针对特定的类型提供专门的模板函数实现。在存在函数特化的情况下,特化版本、模板函数的通用版本以及非模板函数之间的重载关系也需要遵循上述的调用规则。例如:
#include <iostream>

// 非模板函数
void printData(int num)
 {
    std::cout << "非模板函数打印整数: " << num << std::endl;
}

// 模板函数
template<typename T>
void printData(T data) 
{
    std::cout << "模板函数通用版本打印数据: " << data << std::endl;
}

// 模板函数特化,针对char类型
template<>
void printData<char>(char data) 
{
    std::cout << "模板函数特化版本打印字符: " << data << std::endl;
}

int main() 
{
    int num = 10;
    char ch = 'A';
    printData(num);  // 调用非模板函数printData(int)
    printData(ch);  // 调用模板函数特化版本printData<char>(char)
    return 0;
}

在这里,对于 printData 函数,有非模板函数、模板函数通用版本以及针对 char 类型的特化版本。调用 printData(num) 时,根据精确匹配优先原则,会调用非模板函数 printData(int);而调用 printData(ch) 时,由于存在针对 char 类型的特化版本,会优先调用这个特化版本,而不是模板函数的通用版本,同样体现了编译器在选择调用函数时遵循的优先匹配规则。

2. 函数隐藏(Function Hiding)

概念
  • 函数隐藏同样出现在类的继承关系中,是指在派生类中定义了与基类同名的函数(不管参数列表是否相同),此时派生类的函数会隐藏基类中同名的所有函数(包括重载函数),在派生类的作用域内,如果不使用作用域限定符显式指定,就无法访问到基类中被隐藏的同名函数。
    隐藏:在基类和派生类之间发生的关系,函数名相同,派生类的函数把基类的函数给隐藏了,只关注函数数名,不管返回值和参数
示例代码
#include <iostream>

class Base {
public:
    void func() {
        std::cout << "Base类的func函数" << std::endl;
    }

    void func(int num) {
        std::cout << "Base类的func(int)函数" << std::endl;
    }
};

class Derived : public Base {
public:
    void func(double num) {
        std::cout << "Derived类的func(double)函数" << std::endl;
    }
};

int main() {
    Derived derived_obj;
    derived_obj.func(3.0);  // 调用Derived类的func(double)函数

    // 以下代码编译错误,因为Derived类的func函数隐藏了Base类的func函数,
    // 不能直接在Derived类作用域内调用Base类的func函数
    // derived_obj.func();

    // 使用作用域限定符可以访问Base类的func函数
    derived_obj.Base::func();

    return 0;
}

在上述代码中,Derived 类中定义了 func(double num) 函数,它隐藏了 Base 类中的 func 函数和 func(int num) 函数,所以在 Derived 类的作用域内直接调用 func 函数时,编译器会认为是调用 Derived 类自身定义的函数,如果要访问基类中被隐藏的同名函数,需要通过 Base::func() 这样的作用域限定符来明确指定。

特点
  • 存在继承关系:也是基于类的继承场景出现的情况,派生类定义了与基类同名的函数。
  • 同名即隐藏:只要函数名相同就会发生隐藏,与参数列表是否相同无关,而且是隐藏基类中所有同名函数,这一点和重载、覆盖都不同,重载是通过参数差异来区分不同函数,覆盖是严格按照函数签名一致且有虚函数特性来实现的。
  • 作用域相关访问问题:隐藏导致在派生类作用域内,默认情况下无法直接访问基类中被隐藏的同名函数,需要使用作用域限定符来打破这种隐藏效果,才能调用基类的同名函数。

3. 函数覆盖(重写,Function Overriding)

概念
  • 函数覆盖发生在类的继承关系中,是指在派生类中重新定义了基类中的虚函数,并且要求函数签名(函数名、参数列表、返回类型,返回类型如果是指针或引用类型时允许协变)完全一致(除了 const 修饰符,派生类重写函数可以比基类函数多 const 修饰),当通过基类指针或引用调用该函数时,会根据对象的实际类型(是基类对象还是派生类对象)来决定调用基类的函数还是派生类重写后的函数,这是实现多态性的重要机制。

覆盖:覆盖是隐藏的一种特殊情况,派生类和基类的函数,1 函数名相同;2 返回值相同;3 参数列表相同(不包括this指针在内);4 基类函数为虚函数;即为覆盖。
其原理是子类虚函数表的函数地址覆盖了父类虚函数表里函数的指针。
在这里插入图片描述

示例代码
#include <iostream>

class Base {
public:
    virtual void func() {
        std::cout << "Base类的func函数" << std::endl;
    }
};

class Derived : public Base {
public:
    void func() override {
        std::cout << "Derived类的func函数" << std::endl;
    }
};

int main() {
    Base* ptr = new Derived();
    ptr->func();  // 调用Derived类重写后的func函数

    Base base_obj;
    base_obj.func();  // 调用Base类的func函数

    Derived derived_obj;
    derived_obj.func();  // 调用Derived类的func函数

    return 0;
}

在这个例子中,Derived 类重写了 Base 类中的虚函数 func,通过基类指针 ptr 指向派生类对象时,调用 func 函数会执行 Derived 类中重写后的版本,体现了多态性。而直接使用基类对象或者派生类对象调用 func 函数时,则分别调用各自类中定义的函数。

特点
  • 存在继承关系:必须在派生类和基类之间发生,基类中声明了虚函数,派生类对其进行重写。
  • 函数签名要求严格一致:函数名、参数列表、返回类型(遵循协变规则等情况除外)要相同,比如基类函数是 void func(int num),派生类重写的函数也得是 void func(int num),目的是让编译器能准确识别这是重写关系,以便在多态调用时正确执行相应版本的函数。
  • 虚函数特性:基类中的函数必须是虚函数(通过 virtual 关键字修饰),这样编译器才会在运行时根据对象的实际类型来动态决定调用哪个类的函数,实现多态行为。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/952816.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

计算机系统组成(计算机组成原理 基础)

文章目录&#xff1a; 一&#xff1a;体系结构 1.系统组成 1.1 硬件系统 1.2 软件系统 2.工作原理 2.1 冯诺依曼体系 2.2 指令和指令系统 3.性能指标 二&#xff1a;硬件系统 1.主机 1.1 CPU 1.2 内存 2.外设 2.1 外存 2.2 输入设备 2.3 输出设备 2.4 适配器 …

STM32 : 波特率发生器

波特率发生器 1. 发送器和接收器的波特率 波特率寄存器 (BRR): 在串行通信中&#xff0c;发送器和接收器的波特率是由波特率寄存器&#xff08;BRR&#xff09;中的一个值 DIV 来确定的。 2. 计算公式 计算公式: 详细解释 1. 波特率寄存器 (BRR) BRR: 波特率寄存器是一…

全新市场阶段, Plume 生态不断壮大的 RWAfi 版图

加密市场在 2024 年迎来了新的里程碑。BTC 不仅成功推出 ETF&#xff0c;以 BTC 为代表的主流加密货币还在一系列传统金融机构的推动下逐步与主流金融市场接轨。与此同时&#xff0c;随着特朗普成功当选下一任美国总统&#xff0c;他承诺推出一系列友好的加密政策&#xff0c;并…

MySQL的小问题

编码问题 不管官方使用什么编码&#xff1a;latin1、gbk、utf8、utfmb4。统一使用utfmb4 MySQL中的utf8并不是utf-8&#xff0c;它省略了一个字节&#xff0c;只是用三个字节存储所有的符号&#xff0c;utfmb4才是utf-8 远程登录问题&#xff1a; MySQL官方默认没有启动远程…

单片机(MCU)-简单认识

简介&#xff1a; 内部集成了CPU&#xff0c;RAM&#xff0c;ROM&#xff0c;定时器&#xff0c;中断系统&#xff0c;通讯接口等一系列电脑的常用硬件功能。 单片机的任务是信息采集&#xff08;依靠传感器&#xff09;&#xff0c;处理&#xff08;依靠CPU&#xff09;&…

金融项目实战 01|功能测试分析与设计

前置内容&#xff1a;金融项目准备的内容笔记可直接看如下笔记 只看&#xff1a;一、投资专业术语 和 二、项目简介 两部分文章浏览阅读2.3k次&#xff0c;点赞70次&#xff0c;收藏67次。安享智慧理财金融系统测试项目&#xff0c;测试用例&#xff0c;接口测试&#xff0c;金…

vue-cli项目配置使用unocss

在了解使用了Unocss后&#xff0c;就完全被它迷住了。接手过的所有项目都配置使用了它&#xff0c;包括一些旧项目&#xff0c;也跟同事分享了使用Unocss的便捷性。 这里分享一下旧项目如何配置和使用Unocss的&#xff0c;项目是vue2vue-cli构建的&#xff0c;node<20平常开…

5个不同类型的数据库安装

各种社区版本下载官方地址&#xff1a;MySQL :: MySQL Community Downloads 一、在线YUM仓库&#xff08;Linux&#xff09; 选择 MySQL Yum Repository 选择对应版本下载仓库安装包&#xff08;No thanks, just start my download.&#xff09; 下载方法1&#xff1a;下载到本…

《CPython Internals》阅读笔记:p97-p117

《CPython Internals》学习第 7 天&#xff0c;p97-p117 总结&#xff0c;总计 21 页。 一、技术总结 1.词法分析(lexical analysis) 根据《Compilers-Principles, Techniques, and Tools》(《编译原理》第2版)第 5 页&#xff1a;The first phase of a compiler is called …

js逆向说明

一 负载的内容传输用这个格式 Content-Type: multipart/form-data Content-Type 是 HTTP 请求头中的一个字段&#xff0c;它告诉服务器请求体的类型。在这个例子中&#xff0c;Content-Type 的值为 multipart/form-data&#xff0c;这表示请求体采用了 multipart/form-data 格…

什么是负载均衡?NGINX是如何实现负载均衡的?

大家好&#xff0c;我是锋哥。今天分享关于【什么是负载均衡&#xff1f;NGINX是如何实现负载均衡的&#xff1f;】面试题。希望对大家有帮助&#xff1b; 什么是负载均衡&#xff1f;NGINX是如何实现负载均衡的&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源…

spring boot学习第二十三篇:Spring Boot集成RocketMQ

前置条件先安装好RocketMQ 希望在Window10安装rocketMQ并简单使用&#xff0c;可以参考如下文章&#xff1a; Window10安装rocketMQ并简单使用-CSDN博客 1、pom.xml文件里面加上依赖 <dependency><groupId>org.apache.rocketmq</groupId><artifactId&…

OpenCV基础:视频的采集、读取与录制

从摄像头采集视频 相关接口 - VideoCapture VideoCapture 用于从视频文件、摄像头或其他视频流设备中读取视频帧。它可以捕捉来自多种源的视频。 主要参数&#xff1a; cv2.VideoCapture(source): source: 这是一个整数或字符串&#xff0c;表示视频的来源。 如果是整数&a…

使用MATLAB正则表达式从文本文件中提取数据

使用MATLAB正则表达式从文本文件中提取数据 使用Python正则表达式从文本文件中提取数据的代码请看这篇文章使用正则表达式读取文本数据【Python】-CSDN博客 文本数据格式 需要提取 V 后面的数据, 并绘制出曲线. index 1V 0.000000W 0.000000E_theta 0.000000UINV 0.0…

Table-Augmented Generation(TAG):Text2SQL与RAG的升级与超越

当下AI与数据库的融合已成为推动数据管理和分析领域发展的重要力量。传统的数据库查询方式&#xff0c;如结构化查询语言&#xff08;SQL&#xff09;&#xff0c;要求用户具备专业的数据库知识&#xff0c;这无疑限制了非专业人士对数据的访问和利用。为了打破这一壁垒&#x…

怎样使自己处于高能量状态?

在忙碌的生活中&#xff0c;保持高能量状态很关键。以下是一些简单易行的方法。 一、原谅自己&#xff0c;放下过去 别总回想让自己尴尬或犯错的事&#xff0c;这样只会消耗能量。告诉自己“错了就改&#xff0c;别再想”&#xff0c;把精力放在当下和未来。 二、避免内耗&…

Linux高并发服务器开发 第十二天(阻塞/非阻塞 fcntl函数 位图 lseek函数 传入传出参数)

目录 1.阻塞和非阻塞 2.fcntl 函数 3.位图 4.lseek 函数 5.传入参数传出参数 5.1传入参数 5.2传出参数 5.3传入传出参数 1.阻塞和非阻塞 - 阻塞、非阻塞是 设备文件、网络文件具备的属性&#xff08;不是read、write的属性&#xff09;。 - 产生阻塞的场景&#xff1…

【C语言系列】函数递归

函数递归 一、递归是什么&#xff1f;1.1尾递归 二、递归的限制条件三、递归举例3.1举例一&#xff1a;求n的阶乘3.2举例二&#xff1a;顺序打印一个整数的每一位 四、递归与迭代4.1举例三&#xff1a;求第n个斐波那契数 五、拓展学习青蛙跳台问题 一、递归是什么&#xff1f; …

啥!GitHub Copilot也免费使用了

文章目录 前言免费版直接修复代码多文件上下文Agent模式总结 前言 最近&#xff0c;GitHub 给开发者们带来了一个好消息&#xff1a;他们的 AI 编程助手 GitHub Copilot 现在可以免费使用了&#xff01;以前&#xff0c;每个月要花 10 美元才能享受的服务&#xff0c;现在对所…

【OpenGL/Assimp】渲染模型、半透明材质与封装光源

文章目录 渲染成果Assimp库准备&#xff1a;Mesh类修改&#xff1a;透明贴图使用&#xff1a;光源封装&#xff1a;使用方式在如下测试环境中&#xff1a; 渲染成果 Assimp库准备&#xff1a; 从GitHub拉取源码&#xff0c;根据网络教程&#xff0c;借助CMake生成VS工程项目&a…