c++函数模板---3
原创博客:转载请标明出处:http://www.cnblogs.com/zxouxuewei/
模板从大体上,可以分为两种:函数模板和类模板。函数模板是算法库的基础,类模板是建立标准库容器和迭代器的基础。这一小节我们只介绍函数模板。
个人觉得,模板是C++对C的一个非常有力的扩充,即使我们不使用面向对象的机制,仅仅写面向过程的程序,它也是很有用的。因为它可以很大程度上避免了因为参数和函数返回值的类型不确定而引起的函数重载问题。
举一个例子,当我们要编写比大小的函数时,总会碰到这样的问题:函数形参的类型是不确定的:可以使两个int比较,也可以是int和double比 较等等。为了能调用各种类型的比较,唯一的办法就是编写大量的重载函数,让编译器自己判断具体使用哪个。但这样带来的问题很多:首先,必须要写大量的重载 函数;其次对于类型的匹配得非常小心,生怕出错;最后如果你想将大于变成小于,那么你的修改所有的重载函数。而有了模板之后,问题就变得简单多了。
[cpp] view plain copy template <typename T1,typename T2>
int compare(const T1 &v1,const T2 &v2)
{
if(v1 < v2)
return -;
else if(v2 > v1)
return ;
else
return ;
其中,模板以关键字template开头,后接模板形参列表,列表中可以有任意个模板类型形参。模板类型形参在整个模板的作用域中,就能当做某一类型使用 了。整个列表用尖括号括起来。每个模板类型形参前面都要使用class或者typename。二者没有什么本质的区别,typename是后来加入到 C++标准的,一些比较老的程序可能只是用class。后面的定义跟一般的函数类似,不再重复了。
然后我们就可以使用compare函数来比较任何支持“<”操作的类型(不论是内置的,还是自定义)了。
[cpp] view plain copy cout<<compare(,2.3)<<endl;
cout<<compare('a','x')<<endl;
string s1 = "abc";
string s2 = "def";
cout<<compare(s1,s2)<<endl;
下面说几点注意事项:
1.模板类型形参跟一般的形参类似,具有屏蔽外围变量类型的作用:例如:
[cpp] view plain copy typedef double T;
template <typename T>
T foo(const T &val)
{
T tmp = val;
return tmp;
}
2.模板形参的名字不能在模板内部重用:
[cpp] view plain copy typedef double T;
template <typename T>
T foo(const T &a)
{
//typedef double T;
return a;
}
3.我们可以在模板内部定义自己的类型成员:
[cpp] view plain copy //使用迭代器打印容器元素
template <typename T>
void print_by_iter(T& c)
{
typename T::iterator iter = c.begin();
while(iter != c.end())
{
cout<<*iter++<<"\t";
}
}
这里面有两点要注意:首先,调用模板的实参类型必须有iterator。其次,最好在定义类型前加上typenme关键字告诉编译器,你指定的是一 个类型。在我们的例子中,不加的话也是能够顺利执行的,但对于有的情况,就必须加上了,比如:T::size_type * p;
这句代码就有二义性了:如果size_type是一种类型,那么这里应该是声明了指向一个类型的指针的类型;如果它是一个对象,那么这里就是在做乘法运算。而编译器会默认的将它当做对象处理。
4.模板参数不必都是类型,我们也可以定义非类型模板形参。
[cpp] view plain copy //打印数组
template <typename T,size_t N>
void printValue(const T (&parm)[N])
{
for(std::size_t i = ;i != N ; i++)
cout<<parm[i]<<endl;
}
由于函数类型是数组的引用,所以传递给函数的,并不是数组的首地址,而是把数组的类型与元素的个数一起当做一种“类型”。其中非类型模板会被常量代替。当模板内部需要时,就能使用了。
最后是几条关于编写模板的建议:
1.模板的形参使用const引用。因为使用引用,可以避免那些不能复制的类型不能调用模板(比如流),而且如果调用模板的对象比较大,那么可以省去复制的代价。
[cpp] view plain copy //输出到流
template <typename T1,typename T2>
T1& print(T1& s,T2 val)
{
s<<val<<endl;
return s;
}
那么可以使用它向任意流中输出内容:
[cpp] view plain copy double dval = 0.88;
float fval = -12.3;
string oristr = "this is a test",desstr;
ostringstream oss(desstr);
ofstream outFile("result.dat"); print(cout,-);
print(cout,dval);
print(cout,oristr); print(outFile,-);
print(outFile,dval);
print(outFile,fval);
print(outFile,oristr); print(oss,-);
print(oss,dval);
print(oss,fval);
print(oss,oristr); cout<<oss.str()<<endl;
[cpp] view plain copy template <typename T>
int big(T &a,T &b)
{
if (a>b)
return ;
if (a<b)
return ;
}
你只能使用:
[cpp] view plain copy int m = ;
int n = ;
cout<<big(m,n)<<endl;
2.函数体中的测试只用<比较。因为有些类型不一定支持>操作:比如,在我们的比较函数中,使用的是 if(v1<v2)...if(v2<v1),而不是使用if(v1>v2)。其实,这是一种泛型编程的习惯。类似的习惯还有:可以用 “==”或者“!=”判断时,不用<一样。比如在for循环时,最好使用=!来当条件,而不是小于。
c++函数模板---3的更多相关文章
- 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 {…}; 因为是父类与子 ...
随机推荐
- IT公司100题-17-第一个只出现一次的字符
问题描述: 在一个字符串中找到第一个只出现一次的字符.例如输入asdertrtdsaf,输出e. 分析: 最简单的方法是直接遍历,时间复杂度为O(n^2). 进一步思考: 字符串中的字符,只有25 ...
- 大嫂的HTML
<html> <head> <style type="text/css"> *{ margin: 0; padding: 0; ...
- Asp.Net中Ajax实现登陆判断
Default.aspx: <head runat="server"> <title>无标题页</title> <script type= ...
- 多态-II(接口实现)
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...
- c# 只允许一个实例运行
1.单件模式,Singleton,应用程序只能允许一个实例在运行.这是最好的解决方法2.查询系统进程里是不是已经运行.private void Form1_Load(object sender, Ev ...
- JQuery源码分析(二)
立即调用表达式: 任何库与框架设计的第一个要点就是解决命名空间与变量污染的问题.jQuery就是利用了JavaScript函数作用域的特性,采用立即调用表达式包裹了自身的方法来解决这个问题. jQue ...
- My97DatePicker日期控件用法
用法很简单,主要演示都在myDate.html <meta http-equiv="content-type" content="text/html; chars ...
- Windows统一平台: 开发小技巧
Windows统一平台: 开发小技巧 技巧一: 在手机端拓展你应用的显示区域.(WP8.1中也适用) 对于Windows Phone系统的手机, 手机屏幕最上方为系统状态栏(System Tray), ...
- web.config的数据库连接字符串进行加密
连接参考:http://wenku.baidu.com/link?url=nwGug8wxz143A4pvBE_kN6vMU7aF3ojwCKJOyN-TQleZ07iAYrjx_FnFVDOtZAF ...
- var functionName = function() {} vs function functionName() {}
The difference is that functionOne is defined at run-time, whereas functionTwo is defined at parse-t ...