第八章 函数探幽(5)函数模板

(五)函数模板

1.函数模板的使用:属于泛型编程的一种

    函数模板,template <typename AnyType>,指出:建立一个模板,并将类型命名为AnyType。关键字template typename 是必须要有的,除非可以用class来代替typename。可以使用更简单一点的声明,template <class T> function(){}。声明了函数模板之后,我们可以使用T来标识所有的数据类型,这样,当我们调用函数的时候,编译器会根据数据的类型,自动生成相匹配的函数,从而可以简化编程。简而言之,就是需要对多种数据类型使用同一种算法的时候可以使用函数模板。

    函数模板使用的时候,首先要声明template,后面是尖括号,比如template <class T>或者template <typename xx>;然后后面不加任何东西,直接加函数的声明,并且在函数的声明中使用T或xx来作为类型的名称。函数定义部分的书写和声明的书写是一样的,也就是前面的template是不可以省略的。

    函数模板在调用的时候,会根据参数的类型自动生成相应的函数定义,从而简化编程。当需要将同一算法应用于不同类型的时候,请使用函数模板;使用函数模板的时候,如果不需要向前兼容,那么尽量使用关键字typename,而不是使用class。

    使用函数模板的时候,可以像使用常规函数一样,编译器会自动根据所给定的值的类型来生成相应的类型的函数,而不需要像容器那样手动指定类型。也就是说在函数实际运行的时候,并没有函数模板,只有按照函数模板根据不同的类型生成的具体的函数。

2.函数模板的重载

    函数模板重载和函数重载一样,需要有不同的函数特征标,比如引用和指针,或者多个参数,函数模板中,并不是每一个参数都要是模板类型,可以是普通的类型int等

3.模板的局限性

    编写的模板很可能无法处理某些类型,比如,a=b;语句,如果a与b是数组名,便不能成立。再比如a>b表达式,如果a与b是结构,便也不能成立。解决方法有两种,一种是使用运算符重载,另一种更有用的是为特定的类型提供具体化的模板定义

4.显式模板具体化

    对于给定的函数名,可以有非模板函数模板函数和显式具体化模板函数三种类型以及他们的重载版本;具体化优先于常规模板,而非模板又优先于具体化

    显式具体化的原型和定义应该使用template <>来打头,并通过名称来指出类型。应用举例:显示具体化模板函数声明:template <> void swap<job>(job &,job &);其中swap后面的尖括号中的job类型是可选的,也就是可以用template<> void swap(job &,job &);来代替上述声明。显式具体化模板的template后面紧接着的尖括号中是空的,这是与普通模板的区别

5.实例化和具体化

    !!!!!在代码中包含函数模板并不会生成函数定义,它只是一个用于生成函数定义的方案。

    通过模板生成函数定义就叫作模板的实例化,生成函数定义是编译器的工作,模板不能生成函数定义,但是使用int的模板实例会生成函数的定义。这种实例化的方式被称为隐式实例化

    还有一种实例化是显示实例化,意味着可以直接命令编译器来创建特定的实例。显示实例化的意思是“使用模板生成某种特定类型的函数定义”用法如下:template void swap <job> (job &,job &);显式具体化的意思是:“不要使用模板生成函数定义,要使用这个具体的类型生成函数定义”。显式实例化和显式具体化的区别是显式实例化后面没有<>,其他都相同。显式实例化的本质是指定特定的类型为模板生成函数定义,使用的还是原来那个模板函数;而具体化的本质是生成特定类型的函数定义的方案并将它实例化,也就是说使用的模板不再是原来的函数模板,而是为特定的类型专门制定的模板

    还有一种实例化的方式是在程序中创建函数来实现实例化(这也是一种显式实例化),但是这时候参数可能不匹配,因此,需要声明参数的类型,这种方法相当于显式实例化,但是没有使用template关键字。方法如下,比如template <class T> T show(T,T);int x=2;double y=3.22;cout<<show<int>(x,y);这时候参数是两种类型,与模板中要求两个参数是同一种类型不符合,但是我们可以显式地使用模板实例化,来让两个参数都是int类型,这样调用函数的时候double类型的y会被强制类型转换为int。

    显式实例化,隐式实例化,显式具体化被统称为具体化,它们都是对具体类型的函数的定义而不是函数的通用描述,显式实例化和显式具体化的区别是显示实例化template后面没有尖括号。

6.编译器选择使用哪个函数版本——重载解析

(1)也就是同名函数选择的优先级的问题完全匹配,但常规函数优于模板提升转换标准转换用户定义的转换。但完全匹配有时候也有优先级,一般来说越具体,优先级越高(比如int &就比const int &优先级要高,指针也是如此,常量引用或指针与具体的类型相比,优先级要低)。

(2)如果有多个参数,那么选择的复杂程度更高,我们不去深究,这些规则主要是为了产生确定的结果,而不会有歧义。当我们用到这些内容的时候,要注意使表达越清晰越好,而不是功能越多越好。

(3)找出最具体的模板的规则叫作函数模板的部分排序规则(partial ordering rules),什么是越具体呢?说白了就是执行的转换越少越具体,比如T可以表示一个基本类型也可以表示一个指针或引用的时候,编译程序会将T表示为一个基本类型,因为它执行的转换更少,也就是更具体。

(4)自己选择。有时候我们需要自己选择出一个恰当的函数,来代替可能会不正确的按优先级自动选择的函数。比如我们定义了一个模板函数,而且还定义了有具体类型的函数,但此时我们依然希望使用模板函数,要如何操作呢?此时可以用如下的方法:比如lesser是一个模板函数,我可以使用lesser<>(a,b)来使用模板函数的隐式实例化,也可以使用lesser<int>(a,b)来使用模板的显式实例化

7.模板函数的发展

(1)模板函数使用过程中出现的问题:

    第一是并不是总能知道应该声明的类型,比如T,M都是模板类型,则T b;M c;type a=b+c;则a应该使用什么类型呢?我们无法预先做出判断。第二种出现的情况是函数的返回值的类型,由于在函数调用之前,我们不知道函数将使用什么类型的参数,因此函数返回值的类型我们也无法得知,不能来声明。为了解决这两个问题,c++98和c++11分别提出了两个方案,一个是decltype关键字来声明返回类型,另一个是用函数后置返回类型的方法。

(2)关键字decltype(declare type的缩写)

    用法举例:int a;decltype(a) b;则声明了一个和a类型一样的变量b。再比如int x,double y;decltype(x+y) z;则声明了一个和x+y类型相同的变量z。

    采用这种方式就可以在模板函数中声明预先不知道类型的变量了,比如有两个模板类型T和M,可以使用decltype(T+M)xx;来声明一个类型是T+M的变量xx。

    Decltype关键字确定类型的流程:假设有一个decltype(expression)var;声明。则第一步,如果expression是一个不带括号的标识符,则var的类型与此标识符的类型相同,包括const等限定符。第二步,如果expression是一个函数调用,则var的类型与函数的返回值的类型相同(不调用函数,只是去查看原型)。第三步,如果expression是一个用括号括起来的左值,则var的类型是这个左值的引用,比如int a;decltype((a))b;则b是int &类型。第四步,如果前面的都不满足,则var的类型与expression的类型相同,比如expression是一个表达式的情况。

(3)另一种函数声明语法:函数后置返回类型

    c++11新增加了另一种声明和定义函数的语法,也就是返回类型后置声明语法。用来处理不能确定函数返回类型的情况。

    使用关键字auto,auto是一个占位符,并没有实际作用。再加上后面的->类型来声明函数的返回类型,这样可以和decltype配合,从而使模板函数可以适应不同的情况。

    用法举例:template<class T1,class T2> auto gt(T1 x,T2 y)->decltype(x+y){...;return x+y;}此时x和y在参数声明的后面,因此x,y位于作用域内,可以使用它们。

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

推荐阅读更多精彩内容