函数模板

1、函数模板举例

template <typename T>
void Swap(T& arg1, T& arg2)
{
    T temp = arg1;
    arg1 = arg2;
    arg2 = temp;
}

template <typename T>
void Swap(T* arg1, T* arg2, int size)
{
    for (int i = 0; i < size; ++i)
    {
        T temp = arg1[i];
        arg1[i] = arg2[i];
        arg2[i] = temp;
    }
}

int main()
{
    int a = 0, b = 1;
    Swap(a, b);
    int c[10] = { 0 }, d[10] = { 0 };
    Swap(c, d, 10);
    return 0;
}

函数模板支持重载,例子中Swap(a, b)Swap(c, d, 10)调用的函数模板就不一样。

#include <iostream>

template <typename T, typename U>
void Func(T& arg1, U& arg2)
{
    std::cout << arg1 << " TU " << arg2 << std::endl;
}

template <typename U>
void Func(int& arg1, U& arg2)
{
    std::cout << arg1 << " intU " << arg2 << std::endl;
}

int main()
{
    int a = 0, b = 1;
    Func(a, b);                     //0 intU 1

    double c = 1.0, d = 1.1;
    Func<double>(c, d);             //1 TU 1.1
    return 0;
}

上面是另外一种重载函数模板的例子。

2、函数模板显式具体化

模板显式具体化又称特例化,类模板可偏特化(部分特例化)或全特化,但函数模板只能全特化,不能偏特化。
函数模板支持编译器自动推导模板参数类型,但是类模板不支持自动推导。

#include <iostream>

template <typename T>
void Swap(T& arg1, T& arg2)
{
    T temp = arg1;
    arg1 = arg2;
    arg2 = temp;
}

struct MyStruct
{
    int a;
    short b;
    MyStruct(int _a, int _b)
    {
        a = _a;
        b = _b;
    }
    void Show()
    {
        std::cout << "a:" << a << " b:" << b << std::endl;
    }
};

int main()
{
    MyStruct stu1(1, 1);
    MyStruct stu2(2, 2);
    stu1.Show();        //a:1 b:1
    stu2.Show();        //a:2 b:2
    Swap(stu1, stu2);
    stu1.Show();        //a:2 b:2
    stu2.Show();        //a:1 b:1
    return 0;
}

由于模板函数Swap采用了默认模板,如果我们想Swap只交换成员a的值而不交换成员b的值,则需要显式具体化一个MyStruct专属的函数。

#include <iostream>

template <typename T>
void Swap(T& arg1, T& arg2)
{
    T temp = arg1;
    arg1 = arg2;
    arg2 = temp;
}

struct MyStruct
{
    int a;
    short b;
    MyStruct(int _a, int _b)
    {
        a = _a;
        b = _b;
    }
    void Show()
    {
        std::cout << "a:" << a << " b:" << b << std::endl;
    }
};

template<>
//void Swap<MyStruct>(MyStruct& arg1, MyStruct& arg2)       //这两种写法都可以
void Swap(MyStruct& arg1, MyStruct& arg2)
{
    int temp = arg1.a;
    arg1.a = arg2.a;
    arg2.a = temp;
}


int main()
{
    MyStruct stu1(1, 1);
    MyStruct stu2(2, 2);
    stu1.Show();        //a:1 b:1
    stu2.Show();        //a:2 b:2
    Swap(stu1, stu2);
    stu1.Show();        //a:2 b:1
    stu2.Show();        //a:1 b:2
    return 0;
}
函数调用优先级

非模板函数>显式具体化函数>普通模板函数

3、实例化和具体化

代码中包含函数模板本身并不会生成函数定义,它只是一个用于生成函数定义的方案。
编译器使用模板为特定内省生成函数定义时,得到的是模板实例,这种实例化方式称为隐式实例化。

template <typename T>
void Swap(T& arg1, T& arg2)
{
    T temp = arg1;
    arg1 = arg2;
    arg2 = temp;
}

int main()
{
    int a = 0, b = 1;
    Swap(a, b);             //隐式实例化

    double c = 1.0, d = 1.1;
    Swap<double>(c, d);     //显式实例化
    return 0;
}

在声明中使用前缀template<>template区分显式具体化和显式实例化。

显式具体化、隐式实例化、显式实例化统称为模板实例化。
前面讲到函数模板支持编译器自动推导模板参数类型,但是类模板不支持自动推导。
隐式实例化就是让编译器自动推导模板参数类型,而显式实例化就是直接告诉编译器模板参数。

4、缺省模板参数

#include <iostream>

template <typename T = int, typename U>
//template <typename T, typename U = int>       //这两种方法都可以
void Func(T& arg1, U& arg2)
{
    std::cout << arg1 << " " << arg2 << std::endl;
}

int main()
{
    int a = 0, b = 1;
    Func(a, b);                     //0 1

    double c = 1.0, d = 1.1;
    Func<double>(c, d);             //1 1.1
    return 0;
}

函数模板的缺省参数可以是模板参数中任意的参数。

5、非类型模板参数

#include <iostream>

template <typename T = int, typename U, int i>
void Func(T arg1, U arg2)
{
    std::cout << arg1 << " " << arg2 << " " << i <<std::endl;
    U a[i];     //i是常量,所以用来创建数组
}

int main()
{
    int a = 0, b = 1;
    Func<int, int, 10>(a, b);       //0 1 10
    Func<double, int, 5>(1.1, 1);   //1.1 1 5
    return 0;
}

非类型模板参数也可以有默认值。

#include <iostream>

template <typename T = int, typename U, U i = 10>
void Func(T arg1, U arg2)
{
    std::cout << arg1 << " " << arg2 << " " << i <<std::endl;
    U a[i];     //i是常量,所以可被用来创建数组
}

int main()
{
    int a = 0, b = 1;
    Func(a, b);                 //0 1 10
    Func<double, int, 5>(1.1, 1);   //1.1 1 5
    return 0;
}

非类型模板参数的类型比较有限,暂时只有整形、枚举、指针、左值引用、autodecltype等。

6、可变参数模板

变参模板就是一个接受可变数目参数的模板函数或模板类。可变数目的参数被称为参数包。
存在两种参数包:模板参数包,表示零个或多个模板参数;函数参数包,表示零个或多个函数参数。

template <typename T, typename ... Args>
void fun(const T& t, const Args& ... args);

Args是一个模板参数包,表示零个或多个模板类型参数。
args是一个函数参数包,表示零个或多个函数参数。

获取参数包中参数个数
#include <stdio.h>

template <typename T, typename ... Args>
void fun(const T& t, const Args& ... args)
{
    printf("%d %d\n", sizeof ...(Args), sizeof ...(args));
}

int main()
{
    fun(1, 2, 3);       //2 2
    return 0;
}

上述函数参数包就有两个模板类型参数、两个函数参数。

编写可变参数函数模板

可变参数函数通常是递归的,第一步调用处理参数包中的第一个实参,然后用剩余实参调用自身。
为了终止递归还需要定义一个非可变参数的函数,如下代码示例。

#include <iostream>
#include <string>

template <typename T>
std::ostream& myprint(std::ostream& os, const T& t)
{
    return  os << t;
}

template <typename T, typename ... Args>
std::ostream& myprint(std::ostream& os, const T& t, const Args& ... args)
{
    os << t << ", ";
    return myprint(os, args...);
}

int main()
{
    std::string str = "Hello World";
    myprint(std::cout, 1, 2, 3, str);       //1, 2, 3, Hello World
    return 0;
}
转发可变参数模板

这个其实跟递归一个道理,递归就是可变参数模板的自我转发。
下面例子中的myprint2就将参数包转发给了myprint

#include <iostream>
#include <string>

template <typename T>
std::ostream& myprint(std::ostream& os, const T& t)
{
    return  os << t;
}

template <typename T, typename ... Args>
std::ostream& myprint(std::ostream& os, const T& t, const Args& ... args)
{
    os << t << ", ";
    return myprint(os, args...);
}

template <typename T, typename ... Args>
std::ostream& myprint2(std::ostream& os, const T& t, const Args& ... args)
{
    return myprint(os, t, args...);
}

int main()
{
    std::string str = "Hello World";
    myprint2(std::cout, 1, 2, 3, str);      //1, 2, 3, Hello World
    return 0;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,902评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,037评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,978评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,867评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,763评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,104评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,565评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,236评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,379评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,313评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,363评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,034评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,637评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,719评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,952评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,371评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,948评论 2 341