《实用C++》第34课 函数的重载与默认参数

本文转载自: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课视频下载地址:

https://www.cctry.com/thread-289930-1-1.html

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值