C++入门:函数重载

news2024/11/20 13:37:38

目录

一. 函数重载的概念和分类

1.1 什么是函数重载

1.2 函数重载的分类

1.3 关于函数重载的几点注意事项

二. C++实现函数重载的底层逻辑(为什么C++可以实现函数重载而C语言不能)

2.1 编译器编译程序的过程

2.2 为什么C++可以实现函数重载而C语言不能

2.3 Linux环境下C++编译器对函数名的修饰规则


一. 函数重载的概念和分类

1.1 什么是函数重载

函数重载是函数的一种特殊情况,C++允许在同一作用域中定义或声明几个名称相同的函数,这些同名函数经常用于处理形参不同(形参个数、类型或顺序不同),但是实现功能类似的函数。如:int Add(int a, int b)和int Add(double a, double)就可以构成一组函数重载。

注意:C语言不支持函数重载。

1.2 函数重载的分类

函数重载有3种情况:参数类型不同、参数个数不同、参数顺序不同。满足上述3种情况至少其中一种,才能构成函数重载。

参数类型不同

两个重名函数,如果参数类型不同,可以构成函数重载。演示代码1.1定义了两个求加法函数Add,一个是针对整形数据的int Add(int x, int y),另一个是针对双精度浮点型数据的double Add(double x, double y)。在调用Add函数时,如果传入两个整形数据,就调用nt Add(int x, int y),如果传入两个双精度浮点型数据,就调用Add(double x, double y)。

编译器会自动识别调用两个重名函数中的哪一个,无需任何额外语句。

演示代码1.1:

#include<iostream>
using namespace std;

//对整形数据的加法函数
int Add(int x, int y)
{
	cout << "int Add(int x, int y)" << endl;
	return x + y;
}

//对浮点型数据的Add函数
double Add(double x, double y)
{
	cout << "double Add(double x, double y)" << endl;
	return x + y;
}

int main()
{
	int add_int = Add(2, 5);  //整形数据加法
	printf("add_int = %d\n", add_int);  //7

	double add_double = Add(2.1, 3.5);  //浮点型数据加法
	printf("add_double = %lf\n", add_double);  //5.6

	return 0;
}
图1.1 演示代码1.1的运行结果

参数个数不同

演示代码1.2定义了两个Add函数,分别实现对两个数据求加法和对三个数据求加法。通过控制调用函数时传给函数的参数个数,来确定调用哪个Add函数。

演示代码1.2:

#include<iostream>
using namespace std;

//对2个整形数据的加法函数
int Add(int x, int y)
{
	cout << "int Add(int x, int y)" << endl;
	return x + y;
}

//对3个整形数据的加法函数
int Add(int x, int y, int z)
{
	cout << "int Add(int x, int y, int z)" << endl;
	return x + y + z;
}

int main()
{
	int add_two = Add(1, 2);
	printf("add_two = %d\n", add_two);  //3

	int add_three = Add(1, 2, 3);
	printf("add_three = %d\n", add_three);  //6

	return 0;
}
图1.2  演示代码1.2的运行结果

形参顺序不同

演示代码1.3定义了两个func函数,第一个func函数的两个形参char类型数据在前、int类型数据在后,第二个func函数int类型数据在前、char类型数据在前。在调用函数时,通过控制传给函数的参数类型的顺序,来确定调用哪个func函数。

演示代码1.3:

#include<iostream>
using namespace std;

void func(char c, int i)
{
	cout << "void func(char c, int i)" << endl;
}

void func(int i, char c)
{
	cout << "void func(int i, char c)" << endl;
}

int main()
{
	func('A', 5);
	func(5, 'A');
	return 0;
}
图1.3  演示代码1.3的运行结果

1.3 关于函数重载的几点注意事项

返回值不同,不能构成重载

double func(int a)和int func(int a)不构成重载,因为其不满足函数的参数类型、个数或顺序不同中的任意一个,因此不能构成重载。

缺省值不同,不能构成重载

func(int a)和func(int a = 10)不能构成重载,这里的原因与返回值不同时的类似, 不满足函数的参数类型、个数或顺序不同中的任意一个。

总结:判断两个重名函数是否能构成重载,只需看函数参数的个数、类型和顺序是否满足条件,不用关注缺省值、返回值等任何其余问题。

func()和func(int a = 10)可以构成重载

虽然这两个同名函数可以构成重载,但在调用时,会出现歧义。如果通过语句func()调用函数,编译器无法确定这里应当处理为没有传入参数调用func()还是存在缺省参数调用func(10)。 

二. C++实现函数重载的底层逻辑(为什么C++可以实现函数重载而C语言不能)

2.1 编译器编译程序的过程

要弄清楚C++实现函数重载的底层逻辑,首先要清楚编译器编译程序的全过程。将一份程序文献生成可执行文件,要经历编译和链接两段过程,其中编译又可以细分为预编译、编译和汇编三个小过程。汇编过程结束时,每个.c(.cpp)文件都会生成一份目标文件(后缀名为.obj或.o),完成链接过程后,就会生成可执行程序。

图2.1 编译器编译程序的过程

预编译阶段

预编译阶段完成的工作包括:

  • 注释的删除
  • #define定义符号和宏的替换
  • 头文件的包含
  • 条件编译

编译阶段

编译阶段将C++代码或C语言代码转换为汇编代码,完成的工作有:

  • 语法检查
  • 符号汇总(只汇总全局符号,不汇总局部作用域中定义的临时变量名或函数名等)

汇编阶段

将汇编语言转换为计算机能够读懂的机器语言(二进制代码),这个过程完成的具体工作有:

  • 符号表的生成(每个.c/.cpp文件都会生成一份符号表)

符号表中存有符号的名称和其存储在内存中的地址,如果在当前.c\.cpp文件中仅声明了某个符号而没有定义,这样对这个.c/.cpp文件编译时就无法在内存中找到这个符号,这是,符号表中就会存储一份虚拟地址。

链接阶段

链接阶段完成的具体工作有:

  • 合并段表
  • 符号表的合并和重定位

如果在.c文件中调用一个没有被定义的函数或多次被定义的函数,编译器会在链接阶段检测出函数未定义或重复定义。理解链接阶段也是理解为什么C++能够支持函数重载而C语言不能支持函数重载的关键。

2.2 为什么C++可以实现函数重载而C语言不能

假设在主函数中调用Add函数,该函数的函数原型为int Add(int x, int y),而该函数仅被声明而没有被定义。在Windows环境先使用VS2019编译器,分别在C语言和C++编译环境下对程序进行编译,可以观察的报错信息:

  • 在C语言编译环境下,报错信息为:无法解析外部符号_Add
  • C++编译环境下,报错信息为:无法解析外部符号int _cdcel Add(int, int)(?Add@@YAHH@z)
图2.2  C语言报错信息
图2.3  C++报错信息

根据报错信息的不同,可以初步推断,C++编译器在汇总函数名符号时,会对函数名进行修饰。根据初步的推断,我们在VS2019编译器中,定义并调用Add函数(Add函数的定义和调用不再同一个.cpp文件中)。对演示代码2.1进行调试(其中的Add函数已有定义),观察其汇编代码(如图2.4所示),汇编指令 call 表示调用函数,图中Add后面括号里的内容为函数地址。

call指令通俗来讲就是实现“跳转过程”,程序在执行main函数中的命令时,在某一位置跳转去执行被调函数地址处的命令。

对于Add后面括号里的地址在什么阶段填入问题,分以下两种情况讨论:

  • 如果调用Add的.cpp文件中定义了Add函数,那么在编译阶段生成符号表时就会填入函数地址。
  • 如果Add函数定义在了其他.cpp文件而调用Add函数的.cpp文件中仅有函数的声明,那么就需要在链接阶段才会填入函数地址。
图2.4  函数调用的汇报代码

 如果此时发现了两个互相冲突的函数,则无法确定应该执行存储在哪一地址处的函数指令。

  • 对于C语言编译器,在生成符号表时,使用的函数名是原本程序中程序员定义的函数名,根据函数名标识查找函数所在的地址,此时如果存在两个相同的函数,函数名就会发生冲突。
  • 对于C++编译器,生成符号表时使用的函数名是经过一定的修饰规则修饰后的函数名,在函数调用时也是采用经修饰后的函数名标识查找函数所在的地址,只要函数的参数不同,调用两个名称相同的函数就不会存在歧义。

2.3 Linux环境下C++编译器对函数名的修饰规则

Linux环境下C++对函数名的修饰规则为:_Z + 函数名长度 + 函数名 + 参数信息

如:int func()在Linxu环境下被修饰后,函数名变为:_Z4funcv。其中:

  • _Z:表示前缀
  • 4:表示函数名有4个字符
  • func:程序员定义的函数名func
  • v:表示函数没有参数

再比如:int func(int x, int y),经修饰后的函数名变为:_Z4funcii,其中ii表示函数有两个整形参数。对更复杂一些的情况,如int func(int i, int* pi),经修饰后函数名变为:Z4funciPi,其中Pi表示int*类型的参数。

在Window环境下,函数名的修饰规则更为复杂。但是,我们只需要知道C++会按照一定的规则对函数原本的名称进行修饰,通过修饰后的函数名查找函数地址即可,没必要深究修饰规则。

 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

内网安装管家婆软件如何实现外网访问?内网穿透的几种方案教程

管家婆软件从网络架构上分两种版本&#xff1a;web&#xff08;浏览器http端口&#xff09;访问的版本和客户端&#xff08;211固定端口sqlserver数据库&#xff09;访问的版本。公司库管经常用仓库登录管家婆&#xff0c;一旦需要在公司外部登陆访问管家婆客户端&#xff0c;就…

微信中如何接入机器人才比较正常

大家好,我是雄雄,欢迎关注微信公众号:雄雄的小课堂。 前言 为什么会有这个话题?大家都知道最近有个AI机器人很火,那就是AI机器人,关于它的介绍,大家可以自行百度去,我这边就不多介绍了。 好多人嫌网页版玩的不过瘾,就把这个机器人接入到了QQ上,接入到了钉钉上,TG …

Go语言基础知识学习笔记

环境准备 下载安装Golang&#xff1a;https://golang.google.cn/dl/ 因为国外下载速度较慢&#xff0c;我们需要配置国内代理 # 开启包管理工具 go env -w GO111MODULEon # 设置代理 go env -w GOPROXYhttps://goproxy.cn,direct # 设置不走 proxy 的私有仓库&#xff0c;多…

Ajax?阿贾克斯?

一、Ajax简介 AJAX Asynchronous JavaScript and XML&#xff08;异步的 JavaScript 和 XML&#xff09;。 AJAX 不是新的编程语言&#xff0c;而是一种使用现有标准的创新方法。 AJAX 最大的优点是在不重新加载整个页面的情况下&#xff0c;可以与服务器交换数据并更新部分网…

供应链挑战迎刃而解!桑迪亚国家实验室使出“量子杀手锏”

桑迪亚国家实验室的科学家Alicia Magann&#xff08;右&#xff09;&#xff0c;Kenneth Rudinger&#xff08;左上&#xff09;&#xff0c;Mohan Sarovar&#xff08;左下&#xff09;和Matthew Grace&#xff08;未附图&#xff09;开发了基于反馈的量子优化算法&#xff08…

“太极”如何利用混部资源,助力腾讯广告降本增效

编者按&#xff1a;近年来&#xff0c; 随着大模型在NLP领域横扫各种大数据磅单取得巨大成功之后&#xff0c;大数据加大模型成为了AI领域建模的标准范式。搜索、广告、推荐的建模也不例外&#xff0c;动辄千亿参数&#xff0c;上T大小的模型成为各大预估场景的标配&#xff0c…

人脸识别——景联文科技提供3D头模数据采集业务!

“拿起手机刷脸解锁、上下班考勤、支付订单&#xff0c;刷脸已极大地便利了我们的生活。清华大学新闻学院教授沈阳表示&#xff0c;中国人平均每天要暴露在各种摄像头下超过500次。人脸识别已成了我们生活中重要的一部分。由于2D人脸识别容易受到姿态、表情、光照等因素影响&am…

痛苦面具,140天备战阿里巴巴,一个疏忽让我前功尽弃...

面试是走的内推途径&#xff0c;因为内推的简历通过率远高于其他方式;我的内推的途径有&#xff1a;联系我在字节跳动工作的一个大学学长。 在线面试&#xff0c;有个线上文本编辑器&#xff0c;类似leetcode那种&#xff0c;可以在线编程。然而有点紧张&#xff0c;视频面试网…

C#开发的OpenRA游戏加载界面的实现

C#开发的OpenRA游戏加载界面的实现 游戏的UI是一个游戏必备, 但是游戏的UI都是自己处理的,不能使用像Windows自带的UI。 这样游戏的UI,其实也是使用游戏的方式来显示的, 只不过使用了低帧率的方式来显示。 比如OpenRA游戏界面,就会显示如下: 游戏的界面有很多,先从一个简…

2023美赛F题全部代码+数据+结果 数学建模

2023年美赛F题全部思路 数据代码都已完成 全部内容见链接&#xff1a;https://www.jdmm.cc/file/2708700/ 1.根据文献选的GGDP的指标&#xff0c;发现GGDP与水资源等有关&#xff0c;由此可以筛选出影响GGDP的所有因子&#xff0c;并可以用所有因子利用层次分析法建立评价体…

小白指南:手把手教你用低代码开发一个应用页面

一、什么是低代码开发 在了解低代码开发之前&#xff0c;我们先看看使用低代码开发的效果。 低代码开发效果示例 低代码开发是DevEco Studio为HarmonyOS开发者提供的可视化页面的开发方式&#xff0c;具备丰富的UI页面编辑能力&#xff0c;开发者可以在图形化的用户界面上自由拖…

项目质量要怎么保持? 如何借助系统软件进行管理

对于任何项目型的企业总是很关心项目成本的话题&#xff0c;但不知从什么时候开始&#xff0c;高质量等于高成本成了各个企业的一种潜意识。 如果交付的项目产品不符合质量标准&#xff0c;即使企业使用最好的项目管理工具或者每个里程碑都达到并在预算范围内完成项目&#xf…

基于springboot的餐饮管理系统

摘 要本设计以真实场景的菜品信息及相关餐桌预定为设计依据&#xff0c;结合了餐饮管理系统的需求及反映的问题&#xff0c;对系统做出合理的需求分析和界面设计。同时本系统使用了JAVA语言、Springboot框架以及MYSQL作为后台数据库进行开发。从系统描述、系统设计及系统实现等…

【Kubernetes】第二篇 - 购买阿里云 ECS 实例

一&#xff0c;前言 上一篇&#xff0c;简单介绍了 CI/CD 的概念以及 ECS 服务规划&#xff0c;搭建整套服务需要三台服务器&#xff0c;配置如下&#xff1a; ECS 配置启动服务说明2核4GJenkins Nexus Dockerci-server2核4GDocker Kubernetesk8s-master1核1GDocker Kube…

Outcome VS. Output:研发效能提升中,谁会更胜一筹?

2007 年&#xff0c;网景通信公司&#xff08;Netscape&#xff09;的联合创始人 Marc Andreessen 在博客 The Pmarca Guide to Startups 中提出 「Product/Market Fit」 &#xff0c;他写道&#xff0c; 「这意味着在一个良好的市场中&#xff0c;拥有能够满足该市场的产品。」…

SYN480R 解码

目录1.空载情况下2.当有按键被按下3.数据帧分析4.同步码5.数据码6.对24位数据帧分析1.空载情况下 在空载情况下&#xff0c;syn480r 输出引脚&#xff0c;输出的是杂乱无序的波形 2.当有按键被按下 按下按键&#xff0c;会连续输出相同的脉冲波形&#xff0c;放大分析 3.数据…

ParallelsDesktop安装【亲测可行】

我这边安装的是macos最新系统 (Ventura13.2) 本文参考这篇文章安装&#xff0c;但是你完全按照这篇文章会报错&#xff0c;具体可行操作记录如下 一、下载软件和补丁 1、点这里去下载补丁18.0.1 2、点这里去下载对应版本的ParallelsDesktop18.0.1&#xff0c;安装上到试用这里…

RocketMQ 初步了解

RocketMQ 初步了解 前言&#xff1a; ​  近期&#xff0c;因公司使用 RocketMQ 作为消息队列中间件&#xff0c;特此了解。  RocketMQ 是阿里巴巴在 2012 年开发的分布式消息中间件&#xff0c;专为万亿级超大规模的消息处理而设计&#xff0c;具有高吞吐量、低延迟、海量…

计算机图形学:中点BH算法对任意斜率的直线扫描转换方法

作者&#xff1a;非妃是公主 专栏&#xff1a;《计算机图形学》 博客地址&#xff1a;https://blog.csdn.net/myf_666 个性签&#xff1a;顺境不惰&#xff0c;逆境不馁&#xff0c;以心制境&#xff0c;万事可成。——曾国藩 文章目录专栏推荐专栏系列文章序一、问题提出二、…

通俗易懂的机器学习——sklearn鸢尾花分类(KNN)

前言 KNN算法是机器学习中较为简单的入门算法&#xff0c;其主要思想是选取k个与待预测点相近的数据&#xff0c;观察他们的类别&#xff0c;本着离谁近就更像谁的思路对于待预测点进行预测&#xff0c;本文将针对使用sklearn进行KNN算法的使用进行详解 数据预处理 在正式开…