前言
在日常生活中,我们经常会遇到一词多义的情况。我们在C++学习过程中,也会遇到类似的情况:一个函数名代表多个函数,这就是函数重载。
函数重载在我们后续学习中扮演者重要的角色,尤其在自定义类型的学习中起着重要的作用,我们以后在定义自定义类型的构造函数时经常会用到它。
函数重载的概念
函数重载是函数的一种特殊情况,C++允许在同一作用域中声明几个功能类似的同名函数,这些同名函数的形参列表(参数个数 或 类型 或 类型顺序)不同,常用来处理实现功能类似的数据类型不同的问题。
代码示例:
//1、参数个数不同
void func();
void func(int a);
//2、参数类型不同
int Add(int left, int right);
double Add(double left, double right);
//3、参数类型顺序不同
void func(int a, double b);
void func(double a, int b);
函数重载使用的要点
- 函数重载不要滥用,一般只建议功能类似的两个函数进行重载
- 不建议带缺省参数的函数进行重载:某些场景编译器无法识别所调用的函数
- 两个函数只有函数返回值类型不同,不构成函数重载
- C语言不支持函数重载(编译器不支持或者说.c文件不支持)
代码示例:
[代码1] 不建议带缺省参数的函数进行重载:当函数A与函数B构成函数重载,但函数A和函数B有一方或双方的参数为缺省参数时,那么在某些场景下使用它们编译器将无法识别调用的是哪个函数。详细看下面的代码。
//函数A
void fun(int a, int b = 1)
{
cout << a << endl;
cout << b << endl;
}
//函数B
void fun(int a, int b, int c = 1, int d = 1)
{
cout << a << endl;
cout << b << endl;
cout << c << endl;
cout << d << endl;
}
int main()
{
//可识别示例
fun(1);//编译器将调用函数A
fun(1, 2, 3);//编译器将调用函数B
fun(1, 2, 3, 4);//编译器将调用函数B
//不可识别示例
fun(1, 2);//编译无法通过,编译器无法识别调用的A还是B
}
[代码2] 两个函数只有函数返回值类型不同,不构成函数重载
//下面两个函数不构成函数重载
void fun(int a);
int fun(int a);
C++支持函数重载的原理--名字修饰
在C/C++中,一个程序要运行起来需要经历以下几个阶段:预处理、编译、汇编、链接。
在编译阶段,相较于C语言编译器,C++编译器会对函数进行名字修饰,不同的编译器修饰规则不同。以gcc和g++编译器为例:
现有test.c和test.cpp文件,文件中均存在以下两个函数:
int Add(int a, int b)
{
return a + b;
}
void func(int a, double b, int* p)
{ }
采用C语言编译器(gcc)编译后的结果:函数名不变
采用C++编译器(g++)编译后的结果:函数名被修饰为【_Z+函数长度+函数名+类型首字母】
因此,对于同名函数,C++在编译后可进行区分但C语言无法区分。即C++支持函数重载但C语言不支持。
作者言
以上是我对C++中函数重载的理解,若有理解不当的地方,欢迎大家留言指正~