C++函数模板(Template)基础知识讲解
近期有些网友想要了解C++函数模板(Template)基础知识讲解的相关情况,小编通过整理给您分析,同时介绍一下有关信息。
在C++编程中,函数模板(Template)提供了一种强大的工具,使得程序员能够编写出通用的函数代码,从而避免了重复定义相同逻辑的函数。函数模板不仅提高了代码的复用性,还增强了程序的灵活性和可维护性。本文将详细介绍C++函数模板的基础知识,包括其定义、使用、实参推断、重载和特化等内容。
Template 基础篇-函数模板
Template所代表的泛型编程是C++语言中的重要的组成部分,我将通过几篇blog对这半年以来的学习做一个系统的总结,本文是基础篇的第一部分。
为什么要有泛型编程
C++是一门强类型语言,所以无法做到像动态语言(python javascript)那样子,编写一段通用的逻辑,可以把任意类型的变量传进去处理。泛型编程弥补了这个缺点,通过把通用逻辑设计为模板,摆脱了类型的限制,提供了继承机制以外的另一种抽象机制,极大地提升了代码的可重用性。
注意:模板定义本身不参与编译,而是编译器根据模板的用户使用模板时提供的类型参数生成代码,再进行编译,这一过程被称为模板实例化。用户提供不同的类型参数,就会实例化出不同的代码。
函数模板定义
把处理不同类型的公共逻辑抽象成函数,就得到了函数模板。
函数模板可以声明为inline或者constexpr的,将它们放在template之后,返回值之前即可。
普通函数模板
下面定义了一个名叫compare的函数模板,支持多种类型的通用比较逻辑。
templateintcompare(constT&left,constT&right){ if(left (1,2);//使用模板函数
成员函数模板
不仅普通函数可以定义为模板,类的成员函数也可以定义为模板。
classPrinter{ public: templatevoidprint(constT&t){ cout< ("abc");//打印abc
为什么成员函数模板不能是虚函数(virtual)?
这是因为c++ compiler在parse一个类的时候就要确定vtable的大小,如果允许一个虚函数是模板函数,那么compiler就需要在parse这个类之前扫描所有的代码,找出这个模板成员函数的调用(实例化),然后才能确定vtable的大小,而显然这是不可行的,除非改变当前compiler的工作机制。
实参推断
为了方便使用,除了直接为函数模板指定类型参数之外,我们还可以让编译器从传递给函数的实参推断类型参数,这一功能被称为模板实参推断。
如何使用
compare(1,2);//推断T的类型为int compare(1.0,2.0);//推断T的类型为double p.print("abc");//推断T的类型为constchar*
有意思的是,还可以通过把函数模板赋值给一个指定类型的函数指针,让编译器根据这个指针的类型,对模板实参进行推断。
int(*pf)(constint&,constint&)=compare;//推断T的类型为int
当返回值类型也是参数时
当一个模板函数的返回值类型需要用另外一个模板参数表示时,你无法利用实参推断获取全部的类型参数,这时有两种解决办法:
返回值类型与参数类型完全无关,那么就需要显示的指定返回值类型,其他的类型交给实参推断。
注意:此行为与函数的默认实参相同,我们必须从左向右逐一指定。
template
T1sum(T2v2,T3v3){ returnstatic_cast (v2+v3); } autoret=sum (1L,23);//指定T1,T2和T3交由编译器来推断 template T3sum_alternative(T1v1,T2v2){ returnstatic_cast (v1+v2); } autoret=sum_alternative (1L,23);//error,只能从左向右逐一指定 autoret=sum_alternative (1L,23);//ok,谁叫你把最后一个T3作为返回类型的呢? 返回值类型可以从参数类型中获得,那么把函数写成尾置返回类型的形式,就可以愉快的使用实参推断了。
template
autosum(Itbeg,Itend)->decltype(*beg){ decltype(*beg)ret=*beg; for(Itit=beg+1;it!=end;it++){ ret=ret+*it; } returnret; } vector v={1,2,3,4}; autos=sum(v.begin(),v.end());//s=10
实参推断时的自动类型转换
编译器进行模板实参推断时通常不会对实参进行类型转换,只有以下几种情况例外:
普通对象赋值给const引用 int a = 0; -> const T&
数组名转换为头指针 int a[10] = {0}; -> T*
函数名转换为函数指针 void func(int a){...} -> T*
函数模板重载
函数模板之间,函数模板与普通函数之间可以重载。编译器会根据调用时提供的函数参数,调用能够处理这一类型的最特殊的版本。在特殊性上,一般按照如下顺序考虑:
普通函数
特殊模板(限定了T的形式的,指针、引用、容器等)
普通模板(对T没有任何限制的)
对于如何判断某个模板更加特殊,原则如下:如果模板B的所有实例都可以实例化模板A,而反过来则不行,那么B就比A特殊。
templatevoidfunc(T&t){//通用模板函数 cout<<"Ingenericversiontemplate"< voidfunc(T*t){//指针版本 cout<<"Inpointerversiontemplate"<<*t< (&s);//调用指针版本,通过<>告诉编译器我们需要用template而不是普通函数
模板函数特化
有时通用的函数模板不能解决个别类型的问题,我们必须对此进行定制,这就是函数模板的特化。函数模板的特化必须把所有的模版参数全部指定。
template<> voidfunc(inti){ cout<<"Inspecialversionforint"<总结
通过本文的介绍,我们了解了C++函数模板的基本概念及其重要性。函数模板允许我们编写通用的函数代码,适用于多种数据类型,从而大大提高了代码的复用性和灵活性。我们还学习了如何定义和使用函数模板,以及在实参推断、重载和特化等方面的知识。掌握这些基础知识,将有助于我们在实际编程中更好地利用函数模板,提高编程效率和代码质量。
推荐阅读
-
基于PyQt5的HTTP接口测试工具开发实战
-
Java中的URL编码(URLDecoder)与解码(URLEncoder)使用详解
-
Mysql修改root密码的四种方法详解
-
JavaScript中保留两位小数的多种实现方法
-
PHP调用API接口详解:从基础到实践
-
Python中使用PyYAML库来读取、解析和处理YAML文件的方法
近期有些网友想要了解Python中使用PyYAML库来读取、解析和处理YAML文件的方法的相关情况,小编通过整理给您分析,同时介绍...
-
使用Python中的BeautifulSoup (bs4) 解析复杂HTML内容的技巧与示例
-
Microsoft SQL Server 2012 数据库安装图文教程
-
PHP获取本机ip地址实例代码详解
-
C#使用iTextSharp库将图片转换为PDF的步骤及实例代码解析