一、函数模板申明、生成及注意事项
1.1函数模板申明
函数模板可以用来创建一个通用的函数,以支持多种不同的形参,避免重载函数的函数体重复设计。它的最大特点是把函数使用的数据类型作为参数。
函数模板的声明形式为:
template<typename 数据类型参数标识符>
<返回类型><函数名>(参数表)
{
函数体
}
其中,template是定义模板函数的关键字;template后面的尖括号不能省略;typename(或class)是声明数据类型参数标识符的关键字,用以说明它后面的标识符是数据类型标识符。这样,在以后定义的这个函数中,凡希望根据实参数据类型来确定数据类型的变量,都可以用数据类型参数标识符来说明,从而使这个变量可以适应不同的数据类型。
函数模板只是声明了一个函数的描述即模板,不是一个可以直接执行的函数,只有根据实际情况用实参的数据类型代替类型参数标识符之后,才能产生真正的函数。关键字typename也可以使用关键字class,这时数据类型参数标识符就可以使用所有的C++数据类型。
1.2模板函数的生成
函数模板的数据类型参数标识符实际上是一个类型形参,在使用函数模板时,要将这个形参实例化为确定的数据类型。将类型形参实例化的参数称为模板实参,用模板实参实例化的函数称为模板函数。模板函数的生成就是将函数模板的类型形参实例化的过程。
1.3函数模板使用应注意问题
1)函数模板允许使用多个类型参数,但在template定义部分的每个形参前必须有关键字typename或class,即:
template<class 数据类型参数标识符1,…,class 数据类型参数标识符n>
<返回类型><函数名>(参数表)
{
函数体
}
2)在template语句与函数模板定义语句<返回类型>之间不允许有别的语句。如下面的声明是错误的:
1 template<class T> 2 int I; 3 T min(T x,T y) 4 5 { 6 7 函数体 8 9 }
3)模板函数类似于重载函数,但两者有很大区别:函数重载时,每个函数体内可以执行不同的动作,但同一个函数模板实例化后的模板函数都必须执行相同的动作。
二、函数模板的调用
函数模板调用通常有两种调用方式:
1)myswap<float>(a,b);//显示类型调用;
2)myswap(a,b);//自动数据类型推导;
三、函数模板做参数实例
3.1对一个数组进行排序,未使用模板
1 #include<iostream> 2 using namespace std; 3 4 //排序整型数组、字符串数组 5 int mySort(int *arr, int len) 6 { 7 int tmp = 0; 8 if (arr==NULL) 9 { 10 return -1; 11 } 12 for (int i = 0; i < len; i++) 13 { 14 15 for (int j = i+1; j < len; j++) 16 { 17 if (arr[i]<arr[j]) 18 { 19 tmp = arr[i]; 20 arr[i] = arr[j]; 21 arr[j] = tmp; 22 } 23 } 24 } 25 26 return 0; 27 } 28 int myPrint(int *arr,int len) 29 { 30 31 for (int i = 0; i < len; i++) 32 { 33 cout << arr[i] << " " ; 34 } 35 cout << endl; 36 return 0; 37 } 38 void main() 39 { 40 int arr[] = {11,33,4,55,66,78,98}; 41 int size = sizeof(arr) / sizeof(*arr); 42 printf("排序之前 "); 43 myPrint(arr,size); 44 printf("排序之后 "); 45 mySort(arr, size); 46 myPrint(arr,size); 47 system("pause"); 48 }
3.2函数模板做参数实例
1 #include<iostream> 2 using namespace std; 3 4 //排序整型数组、字符串数组 5 template<typename T,typename T2> 6 int mySort(T *arr, T2 len) 7 { 8 T tmp = 0; 9 if (arr==NULL) 10 { 11 return -1; 12 } 13 for (T i = 0; i < len; i++) 14 { 15 16 for (T j = i+1; j < len; j++) 17 { 18 if (arr[i]<arr[j]) 19 { 20 tmp = arr[i]; 21 arr[i] = arr[j]; 22 arr[j] = tmp; 23 } 24 } 25 } 26 27 return 0; 28 } 29 template<typename T, typename T2> 30 int myPrint(T *arr,T len) 31 { 32 33 for (T i = 0; i < len; i++) 34 { 35 cout << arr[i] << " " ; 36 } 37 cout << endl; 38 return 0; 39 } 40 void main() 41 { 42 int arr[] = {11,33,4,55,66,78,98}; 43 int size = sizeof(arr) / sizeof(*arr); 44 printf("排序之前 "); 45 myPrint<int ,int >(arr,size);//显示调用 46 printf("排序之后 "); 47 mySort(arr, size);//自动调用 48 myPrint<int,int>(arr,size); 49 system("pause"); 50 }
四、函数模板与普通函数的差异
1)函数模板不允许自动类型转换;
2)普通函数能够进行自动类型转换。
函数模板可以像普通函数一样被重载,如果一个程序中同时存在函数模板和普通函数,C++编译器有限考虑调用普通函数,如果函数模板可以产出更好的匹配,则选择函数模板。
1 #include<iostream> 2 using namespace std; 3 4 int Max(int a, int b) 5 { 6 cout << "int max(int a,int b)" << endl; 7 return a > b ? a : b; 8 } 9 10 template<typename T> 11 T Max(T a, T b) 12 { 13 cout << "T max(T a,T b)" << endl; 14 return a > b ? a : b; 15 } 16 17 template<typename T> 18 T Max(T a, T b, T c) 19 { 20 cout << "T Max(T a, T b, T c)" << endl; 21 return a > b ? a : b; 22 } 23 void main() 24 { 25 int a = 10; 26 int b = 20; 27 cout << Max(a,b) << endl;//当函数模板和普通函数都符合调用规则,优先选择普通函数 28 cout << Max<>(a, b) << endl;//若显示使用函数模板,则调用函数模板 29 cout << Max(1.0, 2.0, 3.0) << endl;//函数的重载 30 cout << Max('a', 100);//调用普通函数,可以隐式转换 31 system("pause"); 32 }