本文转载自:VC驿站
https://www.cctry.com/thread-289930-1-1.html
1、函数重载的由来:
在平时的编程过程中,一个函数可能就实现一个功能,多个功能就需要多个函数。但有时候多个功能之间很像,只是一些细节有差异,那么类似这样的功能能不能写成一个函数呢?例如,要求2个数中比较大的数,这两个数的类型有时候是int类型,有时候是float类型,那么我们按照之前的写法应该是这样封装函数:
int max_int(int a, int b);
float max_float(float a, float b);
所以,用户在调用的时候就要根据不同的参数类型,调用不同名字的函数,如果要求两个int类型的数中的比较大的就要调用 max_int,如果要求两个float类型的数中的比较大的就要调用 max_float,能不能叫一个名字呢?让系统自己去适配到底调用哪个?答案是可以的。例如:
int max_num(int a, int b);
float max_num(float a, float b);
实际上这就是函数的重载。
C++允许同一函数名定义多个函数,这些函数的参数类型和个数可以不相同,而且至少要有一个不相同,如果都相同的话就会报重复定义的链接错误了。使一个函数名可以多用。
2、函数重载的要求:
重载函数的参数个数、参数类型、参数顺序 三者中必须至少有一种不同(不然会产生调用疑惑)。函数的返回值类型可以相同也可以不同。
单纯返回值类型不同的两个同名函数不能构成函数重载,会报链接错误。
举个例子:
int max_num(int a, int b);
bool max_num(int a, int b);
float max_num(float a, float b);
这样就不行,int max_num(int a, int b); 和 bool max_num(int a, int b); 不满足条件,只有返回值类型不同。这样会导致系统不知道调用哪一个。编译直接报错。
例如:
max_num(2, 5);
这时候如果我问你,要调用哪个,你也不能确定,因为我这里根本就没关心返回值返回什么。所以函数的重载也不是说随便重载。
另外,在跟大家说一下,函数的重载一般都是功能相近,或者功能类似的函数进行重载,不能把一些功能相差很大,或者完全不相关的函数叫同一个名字,语法上是没有错误,但是违背函数重载设计的初衷啦。
3、函数的默认参数:
我们来举个例子,实现一个函数,求出某个int类型数组的最大值和最小值,之前是当做一个小作业给大家出题来着,还有的网友问这个问题,详情见帖子:
https://www.cctry.com/thread-289893-1-1.html
那么现在呢,我想让这个函数更灵活一些,有的时候我只想求最大值,有的时候我只想求最小值,有的时候我同时求最大值和最小值。
void get_min_max(int src[], int arr_len, int* max_v, int* min_v);
能不能不让我每次都得定义两个 int 类型的变量用来接收数组中的最大值和最小值。我需要最大值就给我最大值,我需要最小值就给我最小值。如何才能办到呢?
通过函数的默认参数就可以做到。
修改方法:
函数的声明改成如下:
void get_min_max(int src[], int arr_len, int* max_v = NULL, int* min_v = NULL);
函数的定义改成如下:
void get_min_max(int src[], int arr_len, int* max_v, int* min_v)
{
if (arr_len < 0 || (!max_v && !min_v)) return;
int var_max = src[0], var_min = src[0];
for (int i = 1; i < arr_len; ++i)
{
if (var_min > src[i]) var_min = src[i];
if (var_max < src[i]) var_max = src[i];
}
if (max_v) *max_v = var_max;
if (min_v) *min_v = var_min;
}
之后调用的时候,例如:
int main()
{
int src[] = { 66, 33, 55, 22, 88, 89, 19 };
int max_v = 0;
get_min_max(src, sizeof(src) / sizeof(int), &max_v); //只获取最大值
cout << "最大值是:" << max_v << endl;
int min_v = 0;
get_min_max(src, sizeof(src) / sizeof(int), NULL, &min_v); //只获取最小值
cout << "最小值是:" << min_v << endl;
return 0;
}
具体代码讲解见视频教程!
4、函数默认参数的注意事项:
①、在函数声明的时候指定,如果没有函数的声明则可以放在函数的定义中,但是声明和定义只能选一个;
②、从第一个有默认值的参数开始,后面的所有参数必须都有默认值才行;
③、调用的时候如果要自定义非第一个默认值的参数,那么前面所有的参数都要明确的写上默认值才行;
④、从使用角度来说函数的默认值比重载更方便,从函数内部实现角度来说比函数的重载更复杂。
5、函数的默认参数对函数重载所造成的歧义:
例如,也有个名字叫做 get_min_max 的函数,作用是获取参数给定int整型数组的最大值,并通过函数的返回值返回,函数的代码如下:
//返回int数组中的最大值
int get_min_max(int src[], int arr_len)
{
int var_max = src[0];
for (int i = 1; i < arr_len; ++i)
{
if (var_max < src[i]) var_max = src[i];
}
return var_max;
}
跟刚才的第4知识点中的函数放在一起会有什么情况发生呢?看看vs2013报不报错?为什么呢?具体见视频教程演示!
6、小作业:
实现一个函数,该函数的声明如下:
bool string_upper_diy(char str[], int str_len, bool b_odd_pos = true);
功能为对字符串指定位置的字符变换为大写。
str 参数为字符串的指针;
str_len 为字符串的长度;
b_odd_pos 为true的时候,就要将 str 字符串中的奇数位置的字符变为大写,同时将非奇数位置的字符变为小写;
b_odd_pos 为false的时候,就要将 str 字符串中的奇数位置的字符变为小写,同时将非奇数位置的字符变为大写;
奇数指的是:1,3,5,7,9,11,13,15,17,19...以此类推;
大家明白了吗?动手试着做一做!
第34课视频下载地址: