重载(Overloading)以及模板(Template)

继续《C++ premier plus》的学习

(1)函数重载,通俗来说,就是相同的函数名字名下,存在多个函数,要使得这成立,各个同名函数必须形参列表(也称为“签名”,signature)不同,形参相同,仅仅返回值不同,不能用作函数重载。

(2)如果某个调用,匹配多个函数,编译器是如何选择的?

原则是选择最匹配的(exact match)函数,例如有如下三个函数原型

void stove(double & r1);                          //matches modifiable lvalue

void stove(const double & r2);                 //matches modifiable and non-modifiable lvalue, and rvalue

void stove(double && r3);                       //matches rvalue

并且有如下语句:

double x = 55.5;

const double y = 32.0;

stove(x);                      //使用stove(double & )

stove(y);                      //使用stove(const double &)

stove(x+y);                  //使用stove(double &&)

(3)模板(template)并不是函数的定义,而是告诉编译器如何自动去定义函数的规则,定义两个数据交换的函数模板可以写成如下形式

例子: template <typename T>

void swap(T &a, T &b)

{

     T temp;

      temp = a;

      a = b;

      b = temp;

}

上述代码中,关键词“typename”可以换成“class”,因为C++ 98标准创立以前,使用的是“class”,因此两种写法都是正确的

(4)模板重载

当两个模板名字相同,参数列表不同,即可以重载,方法与函数重载类似,不再赘述,值得提出的是,模板的参数列表允许出现其他类型的具体参数,如

整型,浮点型,等等。

(5)关于优先级

假设一个函数调用匹配了多个函数(exact match),编译器会选择最佳匹配,这个过程称为“重载决策”,如果多个函数中只有一个非模板函数,则选择非模板函数;如果所有函数均为模板函数,选择更具体化(more specialized)的那个;如果有两个或以上同等匹配的非模板函数(模板函数),他们同样具体化,那么编译器将会报错。举例如下,

template <typename T>                                            //模板A

void ShowArray(T arr[], int n);

template <typename T>                                          // 模板B

void Show Array(T * arr[], int n);

有如下语句:

int a[4] = {1, 2, 3, 4};

double * pd[3];

for (int i = 0; i < 3; i++)

      *pd[i] = i;

ShowArray(a, 4);                             //调用模板A

ShowArray(pd, 3);                           //调用模板B

上述例子中,第一次调用ShowArray(),选用模板A,T只用被解释成int 类型即可,因此编译器选择模板A;第二次调用ShowArray(), 选用模板A,T需要被解释成double *类型(指向double类型数据的指针),选用模板B,T只需要被解释成double数据类型,因此B比A更具体化,所以选用模板B

你可能感兴趣的:(overloading)