C++函数模板
函数模板提供了一种函数行为,该函数行为可以用多种不同的类型进行调用,也就是说,函数模板代表一个函数家族,这些函数的元素是未定的,在使用的时候被参数化。
本文地址:http://www.cnblogs.com/archimedes/p/cpp-template.html,转载请注明源地址。
下面举一个简单的例子:
定义模板:
template<typename T>
inline T const& max(T const& a, T const& b)
{
return a < b ? b : a;
}
这个模板定义指定了一个“返回两个值中最大者”的函数家族,这两个值通过函数参数a , b传递给函数模板的,而参数的类型还没有确定,用模板参数T来代替。
其中,typename可以用class来代替,但是建议使用typename
完整代码如下:
#include<iostream>
#include<string>
using namespace std; template<typename T>
inline T const& max(T const& a, T const& b)
{
return a < b ? b : a;
} int main()
{
int i = ;
cout<<"max(7, i): "<<::max(, i)<<endl; double f1 = 3.4;
double f2 = 5.6;
cout<<"max(f1, f2): "<<::max(f1, f2)<<endl; string s1 = "mathematics";
string s2 = "math";
cout<<"max(s1, s2): "<<::max(s1, s2)<<endl;
cin.get();
return ;
}
实参的选择:
模板参数可以根据我们传递的实参来决定,如果我们传递了两个int给参数类型T const&,那么C++编译器得出结论:每个T都必须正确的匹配,如果此时传递的实参为:max(4, 4.2),则出现编译错误
有3种方法来处理上面的错误:
1、对实参进行强制类型转换,使它们可以互相匹配:
max(static_cast<double>(4), 4.2);
2、显示指定(或限定)T的类型:
max<double>(4, 4.2)
3、指定两个参数可以具有不同的类型
模板参数
函数模板有2种类型的参数:
1、模板参数:位于函数模板名称的前面,在一对尖括号内部进行声明:
template<typename T>
2、用参数:位于函数模板名称之后,在一对圆括号内部进行声明:
...max(T const& a, T const& b)
可以声明任意数量的模板参数,在函数模板的内部,不能指定缺省的参数。
template<typename T1, typename T2>
inline T1 max(T1 const& a, T2 const& b)
{
return a < b ? b : a;
}
...
max(, 4.2);
可以引入第三个模板实参类型,来定义函数模板的返回类型:
template<typename T1, typename T2, typename RT>
inline RT max(T1 const& a, T2 const& b);
...
max<int, double, double>(, 4.2);
行得通,但是很麻烦
还有一种方法是只显示的指定第一个实参,而让演绎过程推导出其余的实参。
template<typename RT, typename T1, typename T2>
inline RT max(T1 const& a, T2 const& b);
...
max<double>(, 4.2); //ok, 返回类型是double
重载函数模板
和普通函数一样,函数模板也可以被重载,示例代码如下:
#include<iostream>
#include<string>
using namespace std; inline int const& max(int const& a, int const& b)
{
return a < b ? b : a;
} template<typename T>
inline T const& max(T const& a, T const& b)
{
return a < b ? b : a;
} template<typename T>
inline T const& max(T const& a, T const& b, T const& c)
{
return ::max(::max(a, b), c);
} int main()
{
cout<<::max(, , )<<endl; //调用具有3个函数的模板
cout<<::max(7.9, 34.5)<<endl; //调用max<double>(通过实参演绎)
cout<<::max('a', 'y')<<endl; //调用max<char>(通过实参演绎)
cout<<::max(, )<<endl; //调用int重载的非模板函数
cout<<::max<>(, )<<endl; //调用max<int>(通过实参演绎)
cout<<::max<double>(, )<<endl; //调用max<double>(没有实参演绎)
cout<<::max('a', 23.4)<<endl; //调用int重载的非模板函数 cin.get();
return ;
}
下面的更有用的例子将会为指针和普通的C字符串重载这个求最大值的模板:
#include<iostream>
#include<string>
using namespace std; //求两个任意类型值的最大者
template<typename T>
inline T const& max(T const& a, T const& b)
{
return a < b ? b : a;
} //求两个指针所指向值的最大者
template<typename T>
inline T* const& max(T* const& a, T* const& b)
{
return a < b ? b : a;
} //求两个C字符串的最大者
inline char const* const& max(char const* const& a, char const* const& b)
{
return *a < *b ? b : a;
} int main()
{
int a, b;
a = ;
b = ;
::max(a, b); //max()求两个int值的最大值 string s = "hey";
string t = "you";
cout<<::max(s, t)<<endl; //max()求两个string类型的最大值 int* p1 = &b;
int* p2 = &a;
cout<<::max(p1, p2)<<endl; //max()求两个指针所指向值的最大值 char const* s1 = "Davide";
char const* s2 = "Nico";
cout<<::max(s1, s2)<<endl; //max()求两个c字符串的最大值 cin.get();
return ;
}
以上在所有实现重载里面,都是通过引用来传递每个实例的,但是一般而言,在重载函数模板的时候,最好只是改变那些需要改变的内容,应该把改变限制在以下两个方面:
改变参数的数目或显式地指定模板参数
对于以上的重载,如果改为基于C-string的max()函数,通过传值来传递参数,就不能实现3个参数的max()版本:
#include<iostream>
#include<string>
using namespace std; //求两个任意类型值的最大者(通过传引用进行调用)
template<typename T>
inline T const& max(T const& a, T const& b)
{
return a < b ? b : a;
} //求两个C字符串的最大者(通过传值进行调用)
inline char const* max(char const* a, char const* b)
{
return strcmp(a, b) < ? b : a;
} //求3个任意类型值的最大者(通过传引用进行调用)
template<typename T>
inline T const& max(T const& a, T const& b, T const& c)
{
return max(max(a, b), c);
} int main()
{
cout<<::max(, , )<<endl; //ok const char* s1 = "frederic";
const char* s2 = "anica";
const char* s3 = "lucas";
cout<<::max(s1, s2, s3)<<endl; //error cin.get();
return ;
}
错误在于:如果对3个C-string调用max,则语句return max(max(a, b), c);将产生错误
原因是对于C-string而言,max(a, b)产生了一个新的临时局部值,该值有可能被外面的max函数以传引用的方式返回,将导致传回无效的引用。
C++函数模板的更多相关文章
- c++函数模板作为类的成员函数,编译报错LNK2019的解决方法
为了使某个类的成员函数能对不同的参数进行相同的处理,需要用到函数模板,即template<typename T> void Function(). 编译时报错LNK2019 解决方法: 1 ...
- C++STL - 函数模板
模板主要是为了泛型编程,做到与类型无关 模板有函数模板和类模板,本文主要整理的是函数模板 1.函数模板定义 template<typename 类型形参1,typename 类型形参2,...& ...
- 使用getopt_long来解析参数的小函数模板
getopt_long原型 #define no_argument 0 #define required_argument 1 #define optional_argument 2 struct o ...
- C++函数重载和函数模板
1.函数重载 这是小菜鸟写的一个例子. 函数重载应该注意以下几点: 1.1重载函数有类似的功能: 1.2只能以参数的类型(形参个数和类型)来重载函数, int max(int a,int b);flo ...
- 零值初始化&字符串常数作为函数模板参数
1.在定义一个局部变量时,并希望该局部变量的初始化一个值,可以显示调用其默认构造函数,使其值为0(bool类型默认值为false). template <typename T> void ...
- 让gcc支持成员函数模板的trick
让gcc支持成员函数模板的trick 罗朝辉 (http://www.cnblogs.com/kesalin/) 本文遵循“署名-非商业用途-保持一致”创作公用协议 gcc 4.7.3 不支持成员 ...
- 不可或缺 Windows Native (16) - C++: 函数重载, 缺省参数, 内联函数, 函数模板
[源码下载] 不可或缺 Windows Native (16) - C++: 函数重载, 缺省参数, 内联函数, 函数模板 作者:webabcd 介绍不可或缺 Windows Native 之 C++ ...
- Effective C++ -----条款45:运用成员函数模板接受所有兼容类型
请使用member function templates(成员函数模板)生成”可接受所有兼容类型“的函数. 如果你声明member templates 用于“泛化copy构造”或“泛化assignme ...
- c++ 左值右值 函数模板
1.先看一段代码,这就是一种函数模板的用法,但是红色的部分如果把a写成a++或者写成一个常量比如1,都是编译不过的,因为如果是a++的话,实际上首先是取得a的 值0,而0作为一个常量没有地址.写成1也 ...
- 读书笔记_Effective_C++_条款四十五:运用成员函数模板接受所有兼容类型
比如有一个Base类和一个Derived类,像下面这样: class BaseClass {…}; class DerivedClass : public BaseClass {…}; 因为是父类与子 ...
随机推荐
- Maven更新父子模块的版本号
前置条件: 1.安装有吃饭的家伙JAVA和MAVEN. 首先,需要有一个packaging类型为pom的pom.xml文件即我们的parent项目pom文件.在这个parent项目中配置好groupI ...
- 游戏开发工具之纹理打包器-3.使用GDI+绘图
上一次我们实现了把我们要的图片添加到CTreeCtrl控件里去,并显示图片的缩略图,现在开始我们要讲比较重要的部分--绘图区.为了实现能编辑图片的功能,绘图区应该具有如下功能. 1. 添加删除图片. ...
- Spring3 MVC入门示例
Spring3 MVC 介绍: 1. Spring MVC 是Spring 框架的Web组件,能够开发WEB工程 2. 能与其它框架(Struts2)很好的集成 3. Spring MVC 是以se ...
- Solr官方文档翻译-About & Getting Started
关于(About) 官方文档介绍了所有的Apache Solr实现的重要特性和功能.它是免费的,可以到http://lucene.apache.org/solr/下载. 为了更加的深入和广泛,设计成一 ...
- Spring MVC全局异常处理与拦截器校检
在使用Spring MVC进行开发时,总是要对系统异常和用户的异常行为进行处理,以提供给用户友好的提示,也可以提高系统的安全性. 拦截系统响应错误 首先是拦截系统响应错误,这个可以在web.xml中配 ...
- Cocos2d-x数据存储
分别是使用UserDefault,内置文件管理和sqlite3数据库的一般方式: 主要代码: bool DataScene::init(){ if (!Layer::init()){ return f ...
- UWP开发入门(十四)—— UserControl中Adaptive UI的小技巧
本篇我们通过绘制一个非常简单的UserControl控件,来分享一下对Adaptive UI的理解及一些图形绘制的技巧. 现在流行的APP都少不了精致的用户头像,首先假设我们需要绘制如下的图形作为默认 ...
- 2015百度之星 IP聚合
IP聚合 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) Problem Descri ...
- Winform开发框架之权限管理系统改进的经验总结(2)-用户选择界面的设计
在上篇总结随笔<Winform开发框架之权限管理系统改进的经验总结(1)-TreeListLookupEdit控件的使用>介绍了权限管理模块的用户管理部分,其中主要介绍了其中的用户所属公司 ...
- C#开发的高性能EXCEL导入、导出工具DataPie(支持MSSQL、ORACLE、ACCESS,附源码下载地址)[转]
转自:http://www.cnblogs.com/yfl8910/archive/2012/05/19/2509194.html 作为财务数据核算人员,面对大量的业务与财务数据,借助于传统的EXCE ...