说起函数重载,我不由得想起了C++的“多态”特性。多态又分为静态(编译时)多态和动态(运行时)多态,静态多态即为函数重载,动态多态则是虚函数机制。虚函数水较深,先不讨论,今天我们来看一下函数重载、作用以及要避免的一些坑(尤其是二义性错误)。
一、先来review一下 函数重载的概念
1. 函数名相同;
2. 形参类型和数量不同;
3. 不关返回值的事。
二、如何匹配要使用那个函数呢?
在C++ primary 中文5版 P217我们可以发现端倪。
1. 选定候选函数集 : ① 与被调用函数同名; ② 声明在调用点可见。=》没有,返回无匹配错误
2. 选定可行函数集:考察候选函数的参数列表,找出参数数量相同、且各个参数类型相同或者可以转换为形参的函数;=》没有,返回无匹配错误
3. 选出最佳匹配函数:层层选拔之后,如果幸存者函数有几个的话(不太可能),我们就要找出最最最合适的那一个,怎么找?实参和形参类型最接近的那一个。=》没有,返回无匹配错误
三、函数重载二义性
二义性是指在编译过程中无法找出最匹配的函数,或者说编译器在函数匹配过后还是有多个函数满足要求,无法确定该执行那一个引发的错误。
具体表现:
1. 参数数目引发的歧义
1 int get(){ 2 return 5; 3 } 4 int get(int a = 5){ 5 return a; 6 } 7 //调用get()
//不给参数和有默认参数会造成歧义。
2. 参数隐式转换引发的歧义
1 int get(int m){ 2 return m; 3 } 4 5 long get(long m){ 6 return m; 7 }
//double d = 1.234;
//调用get(d);double既可以隐式转换未long,也可以是int,或者说一般的数值类型之间都可以进行隐式类型转换,故无法确定那一个更加匹配。
1 int get(int a){ 2 return a; 3 } 4 int get(int &a){ 5 return a; 6 } 7 //int m = 10; 8 //调用get(m)
//传值和传引用都可以完成调用,只不过值传递有拷贝的开销,而引用传递不会有。
3. 类型型相关歧义(较深内容,不容易理解)
P517-520,艰涩难懂,需要自己琢磨。
另外,我发现在P517的一个例子,是错误的,并未发生歧义,不会导致运行错误。
1 #include <iostream> 2 #include <string> 3 using namespace std; 4 5 struct B; 6 struct A { 7 A() = default; 8 A(const B&){ 9 //转换B为A 10 cout << "A内部将B转换为A" << endl; 11 }; 12 //... 13 string name = "A"; 14 }; 15 16 struct B { 17 B() = default; 18 operator A() const{ 19 //类类型转换,B->A 20 cout << "B 类类型转换为 A" << endl; 21 }; 22 //... 23 string name = "B"; 24 }; 25 26 A f(const A&) { 27 //... 28 } 29 30 int main() 31 { 32 B b; 33 A a = f(b); 34 }
(由g++运行)
四、如何避免/解决
一是我们要注意默认参数,二是要考虑类型转换,使用explicit可以阻止隐式类型转换。