C++之旅-函数重载

前言

函数重载指的是一个作用域内的几个函数名字相同但是形参列表不同。这些函数执行操作类似,但是接受的形参类型不一样,编译器会根据传递的实参类型选择对应的函数调用。本文将简单介绍C++中的函数重载。

定义重载函数

假设有一个计算图形面积的函数,它可以是计算三角形,圆形或正方形的面积。函数的名字都相同,只是根据传入的图形类型来选择不同的函数来计算面积,程序清单如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
#include <iostream>
using namespace std;
typedef struct Triangle//定义三角形结构
{
double high;//高
double baseLen;//底边长
}Triangle;
typedef struct Circle //定义圆形结构
{
double radius;//半径
}Circle;
typedef struct Square//定义正方形结构
{
double sideLen;//边长
}Square;
//函数1.计算三角形面积
double calcArea(const Triangle&)
{
cout<<"calcute triangle area"<<endl;
}
//函数2.计算圆形面积
double calcArea(const Circle&)
{
cout<<"calcute circle area"<<endl;
return 0;
}
//函数3,计算三角形面积
double calcArea(const Square&)
{
cout<<"calcute square area"<<endl;
}
int main(void)
{
Triangle triangle;
Circle circle;
Square square;
calcArea(triangle);//调用函数1
calcArea(circle);//调用函数2
calcArea(square);//调用函数3
return 0;
}

可以看到,定义的三个函数名calcArea都相同,只是形参类型不同。当分别传入三角形,圆形和正方形类型时,会调用对应的函数。
运行结果如下:

1
2
3
calcute triangle area                                                                                                                                                                       
calcute circle area
calcute square area

可以看到,当分别传入Triangle ,Circle,Square类型时,分别调用了对应的函数。

为什么要重载

函数重载在一定程序上可以减轻程序员起名字的负担。最常见的一个例子就是构造函数的重载。

1
2
3
4
5
6
7
class Test
{
public:
Test(void); // 无参构造函数
Test(int a);//构造函数
Test(int a,int b);//两个整型参数的构造函数
};

可以看到,类Test的三个构造函数名都为Test。如果没有重载,要实现三个构造函数就可能需要三个不同的构造函数名区分,这也就增加了类的使用者的负担,使用者需要传入不同参数构造对象时,就需要使用不同的构造函数名称。而有函数重载之后,类的使用者可以使用同一个函数名传入不同的参数即可。

当然了,如果单纯地为了减轻起名字的负担而去使用函数重载,而使得函数失去了本来的信息,则是一个不明智的选择。我们可以为那些操作确实极其相似的函数进行重载。

不能重载的情况

以下几种情况下,是不能重载或者说是非法的。

main函数不能重载

这是在C++ 11标准中说明的:

1
2
A program shall contain a global function called main, which is the designated start of the program....
This function shall not be overloaded.

试想如果作为用户程序入口函数的main函数被重载了,那么加载的时候该以哪个为入口呢?

只有返回值不同

例如下面两个声明只有返回值不同,函数名和形参都相同:

1
2
3
double calcArea(const Square&);
int calcArea(const Square&); //非法,仅有返回值不同,不可重载
/*以上声明同时出现会报错*/

试想一下,当你传入Square类型参数,而不去使用返回值时,应该调用上面的哪个函数呢?

形参列表看似不同,实则相同

例如使用typedef给Triangle起了一个“别名”:

1
2
3
4
typedef Triangle MyTri;
double calcArea(const Triangle&);
double calcArea(const MyTri&);
/*以上声明同时出现会报错*/

上面这种情况的形参看似不一样,本质上来说它们并没有什么不同。

形参名不同

例如:

1
2
3
4
double calcArea(const Circle &circle );//形参名为circle
double calcArea(const Circle& cir);//形参名为cir
double calcArea(const Circle& );//省略形参名
/*以上声明同时出现会报错*/

这里形参的名字仅仅是起到说明或者记忆的作用,因此对于上面三个声明,它们的形参名可以随意起,但不会影响形参列表的内容。

仅有顶层const的差异

例如:

1
2
3
4
5
6
double calcArea(const Circle);//函数1
double calcArea(Circle);//重复声明了函数1
/*以上声明同时出现会报错*/
double calcArea(Circle* const);//函数2
double calcArea(Circle*);//重复声明了函数2
/*以上声明同时出现会报错*/

但需要特别注意的是,如果形参是指针或引用,是可以通过区分指向大到底是常量对象还是非常量对象来实现函数重载。例如下面的情况是可以实现函数重载的:

1
2
3
4
5
6
double calcArea(const Circle&);//作用于常量引用
double calcArea(Circle&);//
/*以上声明同时出现不会报错*/
double calcArea(const Circle*);//作用于常量指针
double calcArea(Circle*);
/*以上声明同时出现不会报错*/

总结

在定义了重载函数后,我们需要以合理的实参进行调用。大多数情况下,我们很容易判断传入的对应实参需要调用哪个函数,但是有些时候却并不那么容易。我们将会在后面的文章中看到如何进行函数匹配

我们对前面的内容做一个总结:

  • 函数重载能够减轻程序员命名的负担,但这不应该以丢失可读性为代价。
  • main函数不能重载。
  • 重载函数的形参在数量或者类型上要有不同。
  • 不能以返回值作为函数重载要素。
  • C语言没有函数重载。
守望 wechat
关注公众号[编程珠玑]获取更多原创技术文章
出入相友,守望相助!