👦个人主页:@Weraphael
✍🏻作者简介:目前学习C++和算法
✈️专栏:C++航路
🐋 希望大家多多支持,咱一起进步!😁
如果文章对你有帮助的话
欢迎 评论💬 点赞👍🏻 收藏 📂 加关注✨
前言
本章是补充C语言语法的不足,以及C++是如何对C语言设计不合理的地方进行优化的。
一、函数重载的概念
C++允许在 同一作用域中 声明几个功能类似的 同名函数 ,这些同名函数需要满足 形参列表不同;其中形参列表不同包括参数个数不同、类型不同、类型顺序不同。
二、例举重载函数
2.1 参数类型不同
周所周知,C语言是不允许出现同名函数;而C++中的函数重载就可以来处理实现功能类似数据类型不同的问题
2.2 参数个数不同
再来看看以下代码:
首先以上代码构成重载(参数个数不同),只是 无参调用会存在歧义。因为
func()
可以调用void func()
,也可以调用缺省参数void func(int a = 1)
,对于缺省参数,如果没有传参,默认使用缺省参数值。
2.3 参数类型顺序不同
三、分析为什么C++支持函数重载而C语言不支持
为什么C++可以函数重载而C就不可以呢?这就跟 【函数名修饰规则】 有关了。
由于vs
修饰规则过复杂,而Linux
下的gcc
、g++
编译器的修饰规则简单易懂。因此下面用gcc
、g++
来演示过程。(gcc
是Linux
下的c语言编译器,g++
是LInux
下的C++编译器)
- 首先在
Linux
下编写test.c
,其代码内容如下:
- 用
gcc
来编译test.c
文件(gcc
是Linux
下的C语言编译器)
果然在我们的预料中,C语言环境下发生了编译错误。接下来我们可以使用objdump(反汇编)
来查看目标文件或者可执行的目标文件的构成。但首先为了能让源文件编译通过,我们修改文件内容如下:
紧接着用gcc
编译test.c
文件,生成了a.out
的可执行文件,然后再用objdump -S a.out
来查看目标文件或者可执行的目标文件的构成
如下图所示,C语言函数名的存储是直接转化使用函数名,所以如果C语言存在函数重载,那么在调用函数时不知道调用哪个函数,因此不支持函数重载。
- 然后再用
g++
来编译先前的test.c
文件(有函数重载的代码),我们发现是可以编译的过的,生成了a.out
最后,再用objdump -S a.out
来查看目标文件或者可执行的目标文件的构成,如下图所示
C++会因为函数名修饰规则,函数名的存储格式为:_Z + 函数名长度 + 函数名 + 参数类型的首字母
。所以,只要形参列表不同,函数名的存储也就不同。那么在链接时,就不会因函数名的冲突而找不到对应地址。因此,C++是支持函数重载的。
四、总结
- 函数重载需要满足参数个数、类型或者类型顺序不同,注意:其返回值没有要求。
- C语言不支持就是因为直接转化函数名,C++支持是因为函数名修饰规则。