**
C++模板
**
C++提供了两种模板机制:函数模板和类模板
1.函数模板语法
函数模板作用: 建立一个通用函数,其函数返回值类型和形参类型可以不具体制定,用一个虚拟的类型来表示。
语法:
template<typename T>//声明一个模板,告诉编译器后面代码紧跟着T不要报错,T是一个通用数据类型
解释:
template-----声明创建模板
typrname-----表明其后面的符号是一种数据类型,可以用class代替
T—通用的数据类型,名称可以替换,通常为大写 字母
2.函数模板注意事项:
1.自动类型推导,必须推导出一致的数据类型T才可以使用
2.模板必须要确定出T的数据类型,才可以使用
//模板注意事项
template<class T>//typename可以替换成class
void mySwap(T& a, T& b)
{
T temp = a;
a = b;
b = temp;
}
//1.自动类型推导,必须推导出一致的数据类型T才可以使用
void test01()
{
int a = 10;
int b = 20;
char c = 'c';
mySwap(a, b);
//mySwap(a, c);//错误
cout << "a=" << a << endl;
cout << "b=" << b << endl;
}
//2.模板必须要确定出T的数据类型,才可以使用
template<class T>
void func()
{
cout << "func调用" << endl;
}
void test02()
{
//func();//错误,模板不能直接使用,必须确定T的类型
func<int>();//利用显示指定的方式,给T一个类型,才可以使用该模板
}
3.普通函数和模板函数区别:
1.普通函数调用可以发生隐式类型转换
2.函数模板 用自动类型推导, 不可以发生隐式类型
3.函数模板 用显示指定类型,可以发生隐式类型转换
//普通函数
int myAdd01(int a,int b)
{
return a + b;
}
//函数模板
template<class T>
T myAdd02(T a,T b)
{
return a + b;
}
void test01()
{
int a = 10;
int b = 20;
char c = 'c';//c的ASCLII码=99
cout << myAdd01(a, c) << endl;
//自动类型推导
cout << myAdd02(a, c) << endl;//报错,使用自动类型推导时,不会发生隐式类型转换
//显示指定类型
cout << myAdd02<int>(a, c) << endl;
}
}
4.普通函数与函数模板调用规则:
1.如果函数模板和普通模板都可以调用,优先调用普通函数
2.可以通过空模板参数列表 强制调用 函数模板
3.函数模板可以发生函数重载
4.如果函数模板可以产生更好的匹配,优先调用函数模板
例如:
1.调用的是普通函数
void myPrint(int a,int b)
{
cout << "调用的是普通函数" << endl;
}
template<class T>
void myPrint(T a, T b)
{
cout << "调用的是函数模板" << endl;
}
void test01()
{
int a = 10;
int b = 20;
myPrint(a, b);
}
int main()
{
test01();
system("pause");
return 0;
}
2.可以利用空模板参数列表来调用函数模板
例如:
void myPrint(int a,int b)
{
cout << "调用的是普通函数" << endl;
}
template<class T>
void myPrint(T a, T b)
{
cout << "调用的是函数模板" << endl;
}
void test01()
{
int a = 10;
int b = 20;
myPrint<>(a, b);
}
int main()
{
test01();
system("pause");
return 0;
}
3.如果函数模板产生更好的匹配,优先调用函数模板
这里优先调用函数模板
void myPrint(int a,int b)
{
cout << "调用的是普通函数" << endl;
}
template<class T>
void myPrint(T a, T b)
{
cout << "调用的是函数模板" << endl;
}
void test01()
{
char a = 'a';
char b = 'b';
myPrint(a, b);
}
int main()
{
test01();
system("pause");
return 0;
}