C++之函数重载

本文详细介绍了C++中的函数重载概念,包括函数重载的定义、构成条件、编译器调用准则以及注意事项。函数重载允许使用相同的函数名但通过参数个数、类型或顺序来定义不同的函数。在调用时,编译器会根据参数列表匹配合适的函数。然而,重载时需要注意避免默认参数值的冲突和函数指针的匹配问题。此外,C++编译器不能以C的方式处理重载函数,因为这会导致目标名冲突。
摘要由CSDN通过智能技术生成

C++之函数重载

1 函数重载的定义

函数重载:使用同一个函数名定义不同的函数。从本质上来看,就是互相独立的不同函数,每一个函数类型不同。因此,函数重载是由函数名和参数列表决定的。

注意:函数返回值不能作为函数重载的重要依据!

2 构成函数重载的条件

当满足以下三个条件之一时,便可以构成函数重载

  • 函数参数个数不同
// 函数类型:void(int)
void Demo(int x)
{
    printf("x = %d\r\n", x);
}
// 函数类型:void(int, int)
void Demo(int x, int y)
{
    printf("x = %d\r\n", x);
    printf("y = %d\r\n", y);
}
// 函数类型:void(int, int, int)
void Demo(int x, int y, int z)
{
    printf("x = %d\r\n", x);
    printf("y = %d\r\n", y);
    printf("z = %d\r\n", z);
}
  • 函数参数类型不同
void Demo(int x)
{
    printf("x = %d\r\n", x);
}

void Demo(char x)
{
    printf("x = %c\r\n", x);
}
  • 函数参数顺序不同
void Demo(char c, int x)
{
    printf("x = %d\r\n", x);
    printf("c = %c\r\n", c);
}

void Demo(int x, char c)
{
    printf("x = %d\r\n", x);
    printf("c = %c\r\n", c);
}

但是,如果函数的参数类型均相同,仅仅顺序不同同样会出错,如下所示的代码:

void Demo(int x, int y, int z)
{
    printf("x = %d\r\n", x);
    printf("y = %d\r\n", y);
    printf("z = %d\r\n", z);
}

void Demo(int y, int x, int z)
{
    printf("x = %d\r\n", x);
    printf("y = %d\r\n", y);
    printf("z = %d\r\n", z);
}

3 编译器调用重载函数的准则

编译器编译代码的流程:

  1. 将所有同名函数作为候选者

  2. 尝试寻找可行的候选函数

  • 精确匹配实参
  • 通过默认参数匹配实参
  • 通过默认类型转换匹配实参
  1. 匹配成功

如果编译失败的话,有如下两种情况:

  • 找到的候选函数不唯一,出现二义性,失败

  • 无法匹配所有候选者,函数未定义,失败

4 函数重载的注意事项

4.1 避开重载带有指定默认值参数的函数

在我们使用函数重载的过程,要注意避开重载带有指定默认值参数的函数。否则在使用的过程中,会出现二义性,导致编译失败。如下代码所示的错误示例:

void Demo(int x, int y)
{
    printf("x = %d\r\n", x);
    printf("y = %d\r\n", y);
}

void Demo(int x, int y, int z = 0)
{
    printf("x = %d\r\n", x);
    printf("y = %d\r\n", y);
    printf("z = %d\r\n", z);
}

int main()
{
    Demo(1, 2);
    return 0;
}

当对重载函数进行调用时 Demo(1, 2),编译器是无法分辨我们到底是使用 void Demo(int x, int y) 函数,还是使用 void Demo(int x, int y, int z = 0) 函数,因此无法编译通过。

4.2 注意函数重载遇上函数指针

重载函数的名称赋值给函数指针后,当对函数指针进行调用时,将根据下面的方式进行函数匹配

  • 首先,根据重载规则挑选与函数指针参数列表一致的候选者
  • 然后,根据候选者的函数类型与函数指针的函数类型进行匹配

通过如下代码所示的示例进行解释:

typedef int(*PDemo)(int i);

// Demo1
int Demo(int x)
{
    return x;
}
// Demo2
int Demo(int x, int y)
{
    return x * y;
}
// Demo3
int Demo(const char* c)
{
    return strlen(c);
}

int main()
{
    int i = 0;
    PDemo pd = Demo;
    // 一个参数,因此不是Demo1就是Demo3
    // pd的函数类型是int(int)与Demo1相同,因此就是Demo1
    i = pd(1);
    
    return 0;
}

如果将上述示例中函数指针的返回类型由 int 更改成 double,仍通过 pd(1) 进行调用的话,该程序将不能被编译过,因为没有与之匹配的重载函数。

  • 参数列表没有问题
  • 函数返回值类型有问题,因为函数类型包含函数的返回值类型

4.3 C++编译器不能以 C 的方式编译重载函数

由于 C++ 编译器将函数名和参数列表编译成目标名,C 编译器将函数名编译成目标名,这样 C 编译器编译后的重载函数的目标名一致,于是便无法实现重载函数的功能。

  • 6
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值