函数模板

template<class T> //告诉编译器 下面如果出现T不要报错,T是一个通用类型,template<class T>等价于template<typename T>

自动类型推导,必须有参数类型才可以推导

例:

template <class T>
void mySwap(T& a, T& b)
{
    T tmp = a;
    a = b;
    b = tmp;
}
void test()
{
    int a = 10;
    int b = 20;
    //自动推导
    mySwap(a, b);

}

显示指定类型

template <class T>
void mySwap(T& a, T& b)
{
    T tmp = a;
    a = b;
    b = tmp;
}
void test()
{
//指定类型
mySwap<int>(a,b);
}

模板必须要指定出T才可以使用

普通函数和函数模板的区别

普通函数 可以进行隐式类型转换

函数模板不可以进行隐式类型转换

函数模板和普通函数在一起调用规则:

  • 如果出现重载,优先使用普通函数调用,如果普通函数没有实现,报错。
template <class T>
void mySwap(T& a, T& b)
{
    T tmp = a;
    a = b;
    b = tmp;
}

void mySwap(int a, int b)
{
    int tmp = a;
    a = b;
    b = tmp;
}
void test()
{
int a=10;
int b=20;
//模板更好匹配,普通函数需要的是int型数据
mySwap(a,b);
}
  • 可以使用空模板实参列表的语法限定编译时只能通过模板匹配
mySwap<>(a,b);//空参
  • 函数模板可以像普通函数那样可以被重载
  • 如果函数模板可以产生一个更好的匹配,那么选择模板
template <class T>
void mySwap(T& a, T& b)
{
    T tmp = a;
    a = b;
    b = tmp;
}

void mySwap(int a, int b)
{
    int tmp = a;
    a = b;
    b = tmp;
}
void test()
{
char a='a';
char b='b';
//模板更好匹配,普通函数需要的是int型数据
mySwap(a,b);
}

模板实现机制

  • 编译器并不是把函数模板处理成能够处理任何类型的函数
  • 函数模板通过具体类型产生不同的函数
  • 编译器会对函数模板进行两次编译,在声明的地方对模板代码本身进行编译,在调用的地方对参数替换后的代码进行编译

模板的局限性

模板不能解决所有的类型。

如果出现不能解决的类型,可以通过第三代具体化自定义数据类型,解决上述问题

如果具体化能够优先匹配,那么就选择具体化

语法

template<> 返回值 函数名<具体类型>(参数...){}
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容