[C/C++]函数重载,三连问,你会吗?
Posted 慵懒?
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[C/C++]函数重载,三连问,你会吗?相关的知识,希望对你有一定的参考价值。
目录
1. 什么是函数重载
首先,本文是在讲函数重载,那么函数重载是什么呢?
函数重载是指在相同的作用域中,具有相同的名称而形参列表不同的多个函数。***注:与函数返回值无关 ***。
形参列表不同代表:函数的参数个数,参数类型,参数的顺序不同
eg:
我们保证在相同作用域,函数名称相同为func时,参数列表不同表示为以下几种情况:
参数类型不一样:
void func(int val){
}
void func(double val){
}
参数个数不一样
void func(float val){
}
void func(float val,float val1){
}
参数顺序不同
void func(int val,double val1){
}
void func(double val,int val1){
}
返回值不同,会报错
void func(int val,double val1){
}
int func(int val,double val1){
}
在C++中我们可以尝试上面几个函数,放在同一个作用域中,不会出现错误;
而在C文件中,我们只要写了两个函数名称相同的函数,就会报错;
大家可以尝试一下,判断上述结论正确与否。
换而言之,C++支持函数重载,用来处理实现功能类似、数据类型不同的问题。
C语言,不支持函数重载
这也是我们接下来要探讨的问题:
2.为什么C++能够支持函数重载,而C语言不支持函数重载。
简而言之: 函数名的修饰规则不同
我们知道C/C++源文件生成可执行文件的四个过程:
1.预处理:展开头文件/宏替换/去掉注释/条件编译
2.编译:检查语法 ->生成汇编代码
3.汇编 :把汇编代码转化成二进制的机器码 (形成符号表)
4.链接 合成可执行的程序,并对声明 在其他目标文件找到对应的定义
也可以看之前的博客
C语言源代码转变为可执行程序的过程
对于一个C++程序来说,我们在链接阶段会根据函数声明去找到其函数定义,所以对于相同名称的函数,我们在这个阶段需要进行区分;
首先我们介绍两个linux命令
gcc -c test.c -o test.o
g++ -c test.cpp -o. test.o
依据test.c 生成其汇编文件test.o
objdump -S test.o
查看汇编文件test.o的内容
C源文件test.c
#include<stdio.h>
void func(int val){
printf("%d",val);
}
int main(){
func(1);
return 0;
}
其对应的汇编文件:
我们可以看出源文件中的func函数在汇编文件中就是func
我们在看一下同一个源文件以C++修饰方式产生的汇编文件
函数修饰规则
其实我们之前所说的函数修饰规则不同,通过上面两幅图,我们就可以明确的感觉到;
而两者详细的修饰规则为:
gcc函数修饰后名字不变.
g++函数修饰后的名字为Z+函数长度+函数名+参数类型首字母
总结
C++中即使拥有同名函数,经过汇编之后,在链接时能够依靠修饰后的函数名称进行区分;
而C语言无法进行区分,故而C++支持函数重载,C语言不支持函数重载
判断是否是函数重载
func(vector& vec)和func(vector& vec)可以构成函数重载吗?
答案:可以
void func(vector<int>& vec){
}
void func(vector<float>& vec){
}
汇编文件:
func(const int val)和func(int val)可以构成函数重载吗?
答案:不可以
void func(const int val){
}
void func(int val){
}
g++ -c test.c -o test.o
test.c: In function ‘void func(int)’:
test.c:7:6: error: redefinition of ‘void func(int)’
void func(int val){
^
test.c:5:6: error: ‘void func(int)’ previously defined here
void func(const int val){
注:如果本篇博客有任何错误和建议,欢迎伙伴们留言,你快说句话啊!
以上是关于[C/C++]函数重载,三连问,你会吗?的主要内容,如果未能解决你的问题,请参考以下文章
10行Python 代码,实现 AI 目标检测技术,你会吗?