一、函数重载

因为函数重载比较容易理解,并且非常有助于我们理解函数模板的意义,所以这里我们先来用一个经典的例子展示为什么要使用函数重载,这比读文字定义有效的多。

现在我们编写一个交换两个int变量值得函数,可以这样写:

 void swap(int & a, int & b)
{
int tmp; tmp = a;
a = b;
b = tmp; }

假如我们现在想在与上面函数相同的文件中(可以理解为同一个main函数中)交换两个float变量呢,是不是需要重新整一个函数名,重新定义一个函数呢?当然是不需要!

可以直接这样写:

#include <iostream>

void swap(int & a, int & b);
void swap(float & a, float & b); int main()
{
using namespace std; int wallet1 = ;
int wallet2 = ;
float wallet3 = 125.33;
float wallet4 = 235.82; cout << "Before swapping wallet1 and wallet2:" << endl;
cout << "wallet1 = $" << wallet1 << endl;
cout << "wallet2 = $" << wallet2 << endl;
swap(wallet1, wallet2);
cout << "After swapping wallet1 and wallet2:" << endl;
cout << "wallet1 = $" << wallet1 << endl;
cout << "wallet2 = $" << wallet2 << endl; cout << "Before swapping wallet3 and wallet3:" << endl;
cout << "wallet3 = $" << wallet3 << endl;
cout << "wallet4 = $" << wallet4 << endl;
swap(wallet3, wallet4);
cout << "After swapping wallet3 and wallet4:" << endl;
cout << "wallet3 = $" << wallet3 << endl;
cout << "wallet4 = $" << wallet4 << endl; return ;
} void swap(int & a, int & b)
{
int tmp; tmp = a;
a = b;
b = tmp; } void swap(float & a, float & b)
{
float tmp; tmp = a;
a = b;
b = tmp; }

执行结果如下:

由此可以总结出函数重载的相关知识:函数重载就是允许存在多个同名的函数,但是函数的参数列表必须不同,即参数的类型或者数目不一样。如示例中的swap()函数有两个,但是输入的参数类型不一样,编译器就是根据参数列表(也称为函数特征)的不同来确定到底选择哪一个函数去执行!这样我们就可以为相同的操作,但是参数类型不同的函数只用同一个函数名,如示例中都是交换操作,只是参数类型不同而已。虽然函数重载很吸引人,但是很明显看出代码量比较大,所以不要滥用,仅当函数基本上执行相同的任务,但使用不同形式的数据时,才应采用函数重载……

试想一下,我们要交换char, int, short, double, float ...等等这许多变量,如果按照上述方式,正常人都疯了,于是函数模板款款而来……

二、函数模板

首先不说函数模板的定义,待洒家展示一下函数模板的神奇之处先,还是以事实说话,用函数模板实现上述功能:交换两个变量!

 #include <iostream>

 template <typename T>
void Swap(T &a, T &b); int main()
{
using namespace std; int wallet1 = ;
int wallet2 = ;
cout << "Before swapping wallet1 and wallet2:" << endl;
cout << "wallet1 = $" << wallet1 << endl;
cout << "wallet2 = $" << wallet2 << endl;
Swap(wallet1, wallet2);
cout << "After swapping wallet1 and wallet2:" << endl;
cout << "wallet1 = $" << wallet1 << endl;
cout << "wallet2 = $" << wallet2 << endl; double wallet3 = 125.33;
double wallet4 = 235.82;
cout << "Before swapping wallet3 and wallet3:" << endl;
cout << "wallet3 = $" << wallet3 << endl;
cout << "wallet4 = $" << wallet4 << endl;
Swap(wallet3, wallet4);
cout << "After swapping wallet3 and wallet4:" << endl;
cout << "wallet3 = $" << wallet3 << endl;
cout << "wallet4 = $" << wallet4 << endl; char x = 'a';
char y = 'b';
cout << "Before swapping x and y:" << endl;
cout << "x = " << x << endl;
cout << "y = " << y << endl;
Swap(x, y);
cout << "After swapping wallet3 and wallet4:" << endl;
cout << "x = " << x << endl;
cout << "y = " << y << endl;
return ;
} template <typename T>
void Swap(T &a, T &b)
{
T tmp;
tmp = a;
a = b;
b = tmp;
}

执行如下:

这里我们实现了int, double, char类型的交换(但是其他任何类型的两个变量的交换均可实现,有兴趣的同学可以自己试一试,很简单,声明两个变量,调用一下函数就是了),但是可以看到,模板函数的定义就只是这么一小段:

 template <typename T>
void Swap(T &a, T &b)
{
T tmp;
tmp = a;
a = b;
b = tmp;
}

好了,来好好解释一下,缓解同志们迫不及待的心情,嘿嘿……

所谓函数模板,就是让我们编写的函数独立于参数类型,只要函数执行的操作不变,就不用改变代码。函数模板以任意类型的方式来定义函数,以上面的Swap()模板函数为例:第一行指出要建立一个模板,类型为T,关键字template声明这是一个模板,关键字typename声明类型(有些老版本编译器不支持typename,用关键字class代替也一样),这里T是一个标志符,习惯上都用T(当然可以是S,Q,ABC等其他合法标识符),这个T代表着一种类型,但是具体代表什么类型(int?double?char?)是根据编译时输入的参数确定的,如上述main()函数的10,11,15行,编译器发现输入参数是int类型,那么T具体化为int, 同样第30,31,35,T就被具体化为char. 后面的代码就是交换两个变量的值。

要注意的是,函数模板本身并不是函数,它只是告诉编译器如何定义函数(这就是称之为模板的原因,提供一个框架而已),具体的函数是在编译过程中生成的。需要交换int的函数时,编译器按照模板的模式创建这样的函数,用int代替T,即函数是在编译过程中生成的,函数模板并不创建函数,只是创建一个模板去告诉编译器如何生成函数而已,最终的代码(编译之后)不包含任何模板,而只包含了为具体程序生成的实际函数。使用模板的好处是,它使得生成多个函数定义更简单、可靠。

专业一点的表述是,函数模板是通用的函数描述(ps:因为适合任何类型的参数),它们使用泛型来定义函数(ps:T可以带便任何类型),其中泛型可以用具体的类型(如int或者double)替换。通过将类型作为参数传递给模板,可是编译器生成该类型的函数。由于模板允许以泛型(刚刚ps中解释了泛型的感性理解哈)的方式编写程序,因此有时也被称为通用编程。由于类型是用参数表示的,因此模板特性也被称为参数化类型(parameterized types).

C++学习笔记之模板(1)——从函数重载到函数模板的更多相关文章

  1. 小猪猪C++笔记基础篇(六)参数传递、函数重载、函数指针、调试帮助

    小猪猪C++笔记基础篇(六) ————参数传递.函数重载.函数指针.调试帮助 关键词:参数传递.函数重载.函数指针.调试帮助 因为一些事情以及自己的懒惰,大概有一个星期没有继续读书了,已经不行了,赶紧 ...

  2. C++函数重载和函数模板

    1.函数重载 这是小菜鸟写的一个例子. 函数重载应该注意以下几点: 1.1重载函数有类似的功能: 1.2只能以参数的类型(形参个数和类型)来重载函数, int max(int a,int b);flo ...

  3. C++ 函数重载,函数模板和函数模板重载,选择哪一个?

    重载解析 在C++中,对于函数重载.函数模板和函数模板重载,C++需要有一个良好的策略,去选择调用哪一个函数定义(尤其是多个参数时),这个过程称为重载解析. (这个过程将会非常复杂,但愿不要遇到一定要 ...

  4. C++ 函数重载与函数匹配

    <C++ Primer>笔记,整理关于函数重载与函数匹配的笔记. 函数重载 void func(int a); //原函数 void func(double a); //正确:形参类型不同 ...

  5. C++中函数重载和函数覆盖的区别

    C++中经常会用到函数的重载和覆盖,二者也在很多场合都拿出来进行比较,这里我就对二者的区别做点总结: 函数重载: 函数重载指的是函数名相同.函数特征值不同的一些函数,这里函数的特征值指的是函数的参数的 ...

  6. C++ 函数的扩展④--函数重载与函数指针

    //函数扩展--函数重载与函数指针 #include<iostream> using namespace std; //函数参数类型不同 void Fuc(char * b){ print ...

  7. 位运算+引用+const+new/delete+内联函数、函数重载、函数缺省参数

    update 2014-05-17 一.位运算 应用: 1.判断某一位是否为1 2.只改变其中某一位,而保持其它位都不变 位运算操作: 1.& 按位与(双目): 将某变量中的某些位(与0位与) ...

  8. c++中的函数重载、函数重写、函数重定义

    目录 一.函数重载 二.函数重写 三.函数重定义 为了更加深刻的理解 函数重载.重写.重定义,我们可以带着如下这两个问题去思考: 1.子类中是否可以定义父类中的同名成员?为什么? 可以,因为子类与父类 ...

  9. C++学习笔记47:链表的概念与结点类模板

    学堂在线学习笔记 链表的概念与结点类模板 顺序访问的线性群体--链表类 链表是一种动态数据结构,可以用来表示顺序访问的线性群体: 链表是由系列结点组成,结点可以在运行时动态生成: 每一个结点包括数据域 ...

随机推荐

  1. MySQL定期分析检查与优化表

    定期分析表   ANALYZE [LOCAL | NO_WRITE_TO_BINLOG] TABLE tbl_name [, tbl_name]   本语句用于分析和存储表的关键字分布.在分析期间,使 ...

  2. Yii: 设置数据翻页

    一种方法是使用CPagination处理翻页需要的数据如:总数据项数,每页数据项数,当前页,然后在视图中使用CBasePager来绘制. 控制器动作的代码示范: function actionInde ...

  3. Auto Updating the exe from a network location when application starts z

    http://www.codeproject.com/Tips/869588/Auto-Updating-the-exe-from-a-network-location-when?msg=499218 ...

  4. Java并发编程-可重入锁

    可重入锁,也叫做递归锁,指的是同一线程 外层函数获得锁之后 ,内层递归函数仍可以获取该锁而不受影响.在JAVA环境下 ReentrantLock 和synchronized 都是 可重入锁. publ ...

  5. 使用python三方库xlrd解析excel数据

    excel是平常用的比较多的一种数据格式,而在自动化测试过程中,解析其数据以供脚本使用就是一个重要的工作,幸好已有现存的三方库供使用,而不必重新造轮子. 一.安装xlrd模块 到python官网下载h ...

  6. MapReduce 中job.setJarByClass()方法的疑惑

    在调试mr实例的时候,遇到如下的情况,如图所示 说明:就是我的mr程序类名称和我设置的setJarByclass()中设置的不一样,但是程序竟然没有报错!!!!当时把我吓尿了 疑惑:如果这样设置的话, ...

  7. 前端架构:Angular与requirejs集成实践

    这几天angular与requirejs.browserify的集成弄的博主头好晕,今天终于成功集成了requirejs,现写些心得体会在这里. 核心思想:angular加载时有一定的顺序,必须依次加 ...

  8. 一个Web页面的生命周期 ,面试常常被问到

    常规页生命周期阶段 一般来说,页要经历下表概述的各个阶段.除了页生命周期阶段以外,在请求前后还存在应用程序阶段,但是这些阶段并不特定于页.有关更多信息,请参见 ASP.NET 应用程序生命周期概述. ...

  9. 黑马程序员——Objective-c特性

    1. 继承  Objective-c不支持多继承. Super 关键字:调用该类的父类: 超类:父类的另一种说法. 2.自定义NSLog()输出: 在类中添加description方法就可以自定义NS ...

  10. SpringMVC + Spring + MyBatis 学习笔记:提交数据遭遇基础类型和日期类型报400错误解决方法

    系统:WIN8.1 数据库:Oracle 11GR2 开发工具:MyEclipse 8.6 框架:Spring3.2.9.SpringMVC3.2.9.MyBatis3.2.8 使用SpringMVC ...