C++学习之函数模板的使用详解
C++函数模板
template<typename T> void Swap(T &a ,T &b) { T temp; temp = a; a = b; b = temp; }
在使用模板函数时,编译器根据实际的类型生成相应的函数定义。
重载的模板
并非所有的类型都使用相同的算法,可以像重载常规函数那样重载模板函数定义。
template<typename T> void Swap(T &a ,T &b); //#1 template<typename T> void Swap(T *a ,T *b,int n);//#2 最后一个参数是具体类型 int main() { int i =10,j=20; Swap(i,j);//使用#1 const int Lim = 8; int d1[Lim]={0,1,2,3,4,5,6,7}; int d2[Lim]={7,6,5,4,3,2,1,0}; Swap(d1,d2,Lim);//使用#2 } template<typename T> void Swap(T &a ,T &b) { T temp; temp = a; a = b; b = temp; } template<typename T> void Swap(T *a ,T *b,int n) { T temp; for(int i=0;i<n;i++) { temp =a[i]; a[i]=b[i]; b[i]=temp; } }
模板局限性
某些时候,类型T的相应操作只适用于数组,如果T为结构体则模板函数便不成立
同样,如if(a>b)
,如果T为结构,则>便不成立
解决方案:
- 重载运算符号
- 为特定类型提供具体化模板定义
显示具体化
当编译器找到与函数调用匹配的具体化定义时,将使用该定义,不再寻找模板。
- 对于给定的函数名,可以有非模板函数、模板函数和显示具体化模板函数以及各自的重载版本。
- 显示具体化的原型和定义以
template<>
开头,并通过名称来指出类型 - 调用顺序是:非模板函数>具体化模板函数>模板函数
void Swap(job& ,job&); template <typename T> void Swap(T&,T&); template<> void Swap<job>(job& ,job&);//显示具体化 //Swap<job>中<job>是可选的,因为函数的参数类型表明,这是job的一个具体化,所以也可以这样写: template<> void Swap(job& ,job&);
实例化和具体化
注意:函数模板并不会生成函数定义,他只是生成一个用于生成函数定义的方案,编译器使用模板为特定的类型生成函数定义时,得到的是模板实例。
template<typename T> void Swap(T &a ,T &b); int a =10,b=20; Swap(a,b);//因为提供了int类型的参数,所以自动生成了int类型的模板实例。这样是==隐式实例化== //也可以直接命令编译器创建特定的实例 //显示实例化 template void Swap<int>(int &,int &);//使用Swap()模板生成int类型的函数定义 //显示具体化 template<> void Swap<int>(int& ,int&); template<> void Swap(int& ,int&); //区别在于:具体化是不使用Swap()模板函数生成函数定义,而是使用专门为int类型显示定义的函数定义 //简单的理解,具体化是对函数的声明,而实例化是对模板函数的使用
template<typename T> T Add(T a,T b) { return a+b; } int m=6; double x=10.5; Add<double>(x,m); //与Add(x,m)不匹配,因为一个是int一个是double //通过Add<double>实例化,可强制将m转为double //但是同样的对Swap便不能成功,因为Swap中使用的是引用类型 Swap<double>(m,x);//double& 不能指向int
//使用案例 template <typename T> void Swap(T &,T &); template<> void Swap<job>(job&,job&);//具体化 int mian() { template void Swap<char>(char& ,char&); short a,b; Swap(a,b);//隐式实例化 job n,m; Swap(n,m);//显示具体化 char g,h; Swap(g,h);//显示实例化 }
模板函数类型的确定
template<class T1,class T2> void fun(T1 x,T2 y) { ?type? s=x+y; //因为是模板函数,此时?type?类型不确定 }
C++11增加decltype
关键字
template<class T1,class T2> void fun(T1 x,T2 y) { decltype(x+y) s=x+y; //s类型与x+y的类型一致 }
使用decltype(expression) var
的步骤:
1.如果expression没有用括号括起来,则var与expression类型相同,包括const等限定符
double x =5.5; double& z =x; const double* pd; decltype(x) w; //w为double类型 decltype(z) u; //u为double& 类型 decltype(pd) v; //v为const double* 类型
2.如果expression是一个函数调用,则var与返回值类型相同。并不会实际调用函数,编译器通过查看原型来确定返回值类型
3.如果expression是一个左值,则var为指向其类型的引用。常见的情况如下:
double x = 4.5; decltype((x)) r = x;//r是double&类型 decltype(x) r = x;//r是double类型 //括号不会改变expression的值和左值性 //可理解为加括号仅仅是decltype声明引用的一种方式
4.如果前3条都不满足,则var与expression类型相同
int j=3; int &k=j; int &n=j; decltype(j+6) x; //x是int decltype(k+n) y;//y是int ,虽然k和n是引用,但是k+n不是引用是2个int的和
如果多次声明,可以结合typedef
和decltype
typedef decltype(x+y) xytype; xytype z = x+y; xytype arr[10];
但是某些需定义返回值类型的函数模板任然不能得到解决,如:
template<class T1,class T2> ?type? fun(T1 x,T2 y) //此时无法确定类型 { return x+y; }
C++新增语法auto h(int x,float y) -> double
,这称为后置返回类型,auto是一个占位符
template<class T1,class T2> auto fun(T1 x,T2 y)->decltype(x+y) //后置类型使用decltype { return x+y; }
到此这篇关于C++学习之函数模板的使用详解的文章就介绍到这了,更多相关C++函数模板内容请搜索猪先飞以前的文章或继续浏览下面的相关文章希望大家以后多多支持猪先飞!
原文出处:https://www.cnblogs.com/qsnn/p/16370860.html
相关文章
- eval函数在php中是一个函数并不是系统组件函数,我们在php.ini中的disable_functions是无法禁止它的,因这他不是一个php_function哦。 eval()针对php安全来说具有很...2016-11-25
- 在php中eval是一个函数并且不能直接禁用了,但eval函数又相当的危险了经常会出现一些问题了,今天我们就一起来看看eval函数对数组的操作 例子, <?php $data="array...2016-11-25
- vector是表示可以改变大小的数组的序列容器,本文主要介绍了C++STL标准库std::vector的使用详解,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2022-03-06
Python astype(np.float)函数使用方法解析
这篇文章主要介绍了Python astype(np.float)函数使用方法解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下...2020-06-08- 这篇文章主要介绍了C++中取余运算的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-02-23
- 这篇文章主要介绍了Python中的imread()函数用法说明,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-03-16
- 这篇文章主要介绍了C++ string常用截取字符串方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-04-25
- 本文通过例子,讲述了C++调用C#的DLL程序的方法,作出了以下总结,下面就让我们一起来学习吧。...2020-06-25
- 本文主要介绍了C# 中取绝对值的函数。具有很好的参考价值。下面跟着小编一起来看下吧...2020-06-25
- 下面小编就为大家带来一篇C#学习笔记- 随机函数Random()的用法详解。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2020-06-25
- 本篇文章主要介绍了C++中四种加密算法之AES源代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。...2020-04-25
- 整数拆分,指把一个整数分解成若干个整数的和。本文重点给大家介绍C++ 整数拆分方法详解,非常不错,感兴趣的朋友一起学习吧...2020-04-25
- CREATE FUNCTION ChangeBigSmall (@ChangeMoney money) RETURNS VarChar(100) AS BEGIN Declare @String1 char(20) Declare @String2 char...2016-11-25
Android开发中findViewById()函数用法与简化
findViewById方法在android开发中是获取页面控件的值了,有没有发现我们一个页面控件多了会反复研究写findViewById呢,下面我们一起来看它的简化方法。 Android中Fin...2016-09-20- 这篇文章主要介绍了C++中Sort函数详细解析,sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变...2022-08-18
- strstr() 函数搜索一个字符串在另一个字符串中的第一次出现。该函数返回字符串的其余部分(从匹配点)。如果未找到所搜索的字符串,则返回 false。语法:strstr(string,search)参数string,必需。规定被搜索的字符串。 参数sea...2013-10-04
- 这篇文章主要介绍了C++万能库头文件在vs中的安装步骤(图文),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-02-23
PHP函数分享之curl方式取得数据、模拟登陆、POST数据
废话不多说直接上代码复制代码 代码如下:/********************** curl 系列 ***********************///直接通过curl方式取得数据(包含POST、HEADER等)/* * $url: 如果非数组,则为http;如是数组,则为https * $header:...2014-06-07- Foreach 函数(PHP4/PHP5)foreach 语法结构提供了遍历数组的简单方式。foreach 仅能够应用于数组和对象,如果尝试应用于其他数据类型的变量,或者未初始化的变量将发出错误信息。...2013-09-28
- free函数是释放之前某一次malloc函数申请的空间,而且只是释放空间,并不改变指针的值。下面我们就来详细探讨下...2020-04-25