如果同一作用域的几个函数名相同但形参列表不同,我们称之为重载函数。这些函数接受的参数类型不同,但是执行的操作非常类似。当调用这些函数时,编译器会根据传递的实参类型推断想要的是哪个函数。函数重载可以在一定程度上减轻程序员起名字,记名字的负担。main 函数不能重载。
不允许两个函数除了返回类型外其他所有的要素都相同:
1 int gel(int a, int b){} 2 3 char gel(int a, int b, int c){}//在参数列表不同的情况下允许返回类型不同 4 5 // char gel(int a, int b){}//错误,不允许两个函数除了返回类型外其他要素都相同 6 7 int gel(char a, char b){}
判断两个形参的类型是否相异:
有时候两个形参列表看起来不一样,但实际上是相同的:
1 //下面每对声明的是同一个函数 2 int gel(const int &x); 3 int gel(const int&);//省略了形参名字 4 5 typedef int phone;//phone是int的别名 6 int lou(const phone&); 7 int lou(const int&);
还有前面的博客中说过,在形参中,顶层 const 是可以忽略的。这点在函数重载里面也是一样的:
1 //这四条声明语句是两两等价的 2 int gel(const int a); 3 int gel(int a); 4 5 int lou(int* const a); 6 int lou(int*); 7 8 //下面两条声明语句只有返回值不同,无法通过编译 9 int ting(const int a); 10 double ting(int a);
调用重载函数:
定义了一组重载函数后,我们需要以合理的实参调用它们。函数匹配是指一个过程,在这个过程中我们把函数调用与一组重载函数中的某个关联起来,函数匹配也叫重载确认。编译器首先将调用的实参与重载集合中每个函数的形参进行比较,然后根据比较的结果决定到底调用哪个函数。一般情况下这个过程是比较容易判断的,但有一些情况下比较难看出最终的选择函数。比如:当两个重载函数形参数量相同且可以相互转换时。
调用重载函数时有三种可能的结果:
1.编译器找到一个实参最佳匹配的函数,并生成调用该函数的代码。
2.找不到任何一个函数与调用的实参匹配,此时编译器发出无匹配的错误信息。
3.有多于一个函数可以匹配,但是每种都不是最佳匹配。此时也将发生错误,称为二义性错误。
重载与作用域:
和一般变量作用域规则一样,如果我们在内层作用域中声明名字,它将隐藏外层作用域中声明的同名实体。在不同的作用域中无法重载函数名。事实上,一般来说,将函数声明置于局部作用域内不是一个明智的选择。