C++函数参数匹配规则
阅读原文时间:2023年08月18日阅读:2

C++ 函数参数匹配

void f(); //f1
void f(int); //f2
void f(int, int); //f3
void f(double, double=3.14);//f4

int main() {
    f(5.6); //调用f4
    return 0;
}

candidate functions:函数名称相同(f1, f2, f3, f4 都是)。

viable functions:参数个数相同(排除f1, f3),且参数可以转换成相同类型(f2, f4都是viable function)。如果不存在viable functions,则编译器报参数不匹配错误(可以通过linting检查)。 最后决定参数类型是否匹配,如果匹配优先调用,不能则选择可以隐式转换成类型相同的函数。

void f(); //f1
void f(int); //f2
void f(int, int); //f3
void f(double, double=3.14);//f4

int main() {
    f(42, 5.6); //报错,参数模糊
    return 0;
}

condidate functions: f1, f2, f3, f4

viable functions: f3, f4

优先级: 精确匹配的参数个数越多优先级越高,参数个数相同优先级相同,如果存在多个最高优先级的函数,则报参数模糊错误。

优先级:

  1. 精确匹配:包括类型相同, 数组和参数名转换为指针,忽略顶层const
  2. const 转换 (把非const实参传给const形参)
  3. promotion数据提升,如int->long, char->unsigned等
  4. 算术转换或者指针类型转换
  5. 类类型(class-type)转换,如string->bool

Promotion and Arithmetic Conversion

void ff(int); //f1
void ff(short); //f2
void manip(long); //f3
void manip(float); //f4

int main() {
    ff('a'); //调用f1,char -> int是promotionO(比int短的数据类型统一promotion为int,
             // char->short 是算术转换
    manip(3.14); //error: ambiguous call,3.14视为double,double->float和doule->long在优先级上是等价的
    return 0;
}

const Arguments

忽略顶层const, 原因是传参的时候实际上进行的是copy过程,即copy一份实参给形参,copy会忽略顶层const

void f(int a);
void f(const int a);//报错,重复定义,两者只能定义一种
void f(int *a);
void f(int* const a);//报错,重复定义,两者只能定义一种

const 转换

void f(int &); //f1
void f(const int &);//f2

int main() {
    const int a{0};
    int b{0};
    f(a); //调用f2(精确匹配)
    f(b); //调用f1(没有定义f1时,调用f2)
    return 0;
}

另外,

void f(int);
void f(int &);

int main() {
    int i = 0;
    f(i); //报错,ambiguous call
    return 0;
}