c++ 函数模板_C++函数模板(泛型编程)
模板是泛型编程的基础,泛型编程即以一种独立于任何特定类型的方式编写代码。
模板是创建泛型类或函数的蓝图或公式。库容器,比如迭代器和算法,都是泛型编程的例子,它们都使用了模板的概念。
每个容器都有一个单一的定义,比如 向量,我们可以定义许多不同类型的向量,比如 vector 或 vector 。
您可以使用模板来定义函数和类,接下来让我们一起来看看如何使用。
模板的引入
之前我们知道的交换两个变量的方法有宏定义、函数,这两种方式都能实现两个变量的交换,但是各有各的优缺点
宏定义:#define SWAP(a,b) {a=b-a;b=b-a;a=a+b;}
- 优点:代码复用,适合所有的类型
- 缺点:缺少类型检查,宏在预处理阶段就被替换掉,编译器并不知道宏的存在
函数:
- 优点:真正的函数调用,编译器对类型进行检查
- 缺点:类型不同需要重复定义函数,代码无法复用
上边两种方式都各有利弊,但是在C++中,存在泛型编程的概念:即不考虑具体数据类型的编程方式(如下)
如何定义函数模板
C++中的泛型编程有函数模板与类模板,这章我们先来了解函数模板
函数模板是一种特殊的函数,可以使用不同的类型进行调用,对于功能相同的函数,不需要重复编写代码,并且函数模板与普通函数看起来很类似,区别就是类型可以被参数化
函数模板通过template与typename(或者class)两个关键字来定义,如下
上边就定义了一个变量交换的函数模板,在使用函数模板时有两种方式
- 自动类型推到调用 Swap(a, b)
- 具体类型显示调用 Swap(a, b)
下边以代码来体验一下函数模板:
#includeusing namespace std;template <class T>void Swap(T& a, T& b){ T tmp = a; a = b; b = tmp;}void main(){ int a = 10; int b = 20; Swap(a, b); //自动推到调用 //Swap(a, b);//显示指定调用 cout <"a = " <" b = " <endl; float c = 12.3; float d = 23.4; //Swap(c, d); //自动推到调用 Swap<float>(c, d); //显示指定调用 cout << "c = " << c << " d = " << d << endl; system("pause");}
可以看到,我们使用函数模板,根据具体类型的参数化,就能适用于不同类型的变量交换,达到了代码复用的效果。
下边来深入理解下函数模板:
- 对于函数模板中使用的类型不同,编译器会产生不同的函数
- 编译器会对函数模板进行两次编译
- 第一次是对函数模板本身进行编译,包括语法检查等
- 第二次是对参数替换后的代码进行编译,这就相当于编译普通函数一样,进行类型规则检查等。
需要注意的是
- 函数模板是不允许隐式类型转换的,调用时类型必须严格匹配
传入不同类型会提示错误。
定义多个类型参数
函数模板还可以定义任意多个不同的类型参数,但是对于多参数函数模板:
- 编译器是无法自动推导返回值类型的
- 可以从左向右部分指定类型参数
#include using namespace std;//class typename都可以使用template <class T1, class T2, typename T3>T1 add(T2 a, T3 b){ T1 ret; ret = static_cast(a + b); return ret;}void main(){ int c = 12; float d = 23.4; //cout << add(c, d) << endl; //error,无法自动推导函数返回值 cout << add<float>(c, d) << endl; //返回值在第一个类型参数中指定 cout << add<int, int, float>(c, d) << endl; system("pause");}
在上边的代码中,我们定义了多类型参数的函数模板,调用时需要注意的是函数返回值需要在第一个参数类型中显示指定,后边的类型可自动推导或显示指定。
函数模板也可以重载
函数模板跟普通函数一样,也可以被重载
- C++编译器优先考虑普通函数
- 如果函数模板可以产生一个更好的匹配,那么就选择函数模板
- 也可以通过空模板实参列表<>限定编译器只匹配函数模板
#include using namespace std;template void fun(T a){ cout << "void fun(T1 a)" << endl;}template void fun(T1 a, T2 b){ cout << "void fun(T1 a, T2 b)" << endl;}void fun(int a, float b){ cout << "void fun(int a, float b)" << endl;}void main(){ int a = 0; float b = 0.0; fun(a); fun(a, b); //普通函数void fun(int a, float b)已经能完美匹配,于是调用普通函数 fun(b, a); //这个调用,函数模板有更好的匹配,于是调用函数模板 fun<>(a, b); //限定只使用函数模板 system("pause");}
从输出可以得到,编译器会优先去调用普通函数,但是当函数模板有更好的匹配时或使用限定符<>时,编译器就会去匹配函数模板。
总结
- 函数模板是泛型编程在C++中的应用方式之一
- 函数模板能够根据实参对参数类型进行推导
- 函数模板支持显示的指定参数类型
- 函数模板是C++中重要的代码复用方式
- 函数模板通过具体类型产生不同的函数
- 函数模板可以定义任意多个不同的类型参数
- 函数模板中的返回值类型必须显示指定
- 函数模板可以像普通函数一样重载
公众号ID:C语言Plus扫码关注最新动态
c++ 函数模板_C++函数模板(泛型编程)相关推荐
- java map函数指针_C 函数指针与回调函数
关于回调函数的一点思考 在 A 函数中将参数与 C 函数传递给 B 函数,B函数调用 C 函数:B 函数调用的动作称为回调,C 函数称为回调函数. 举栗 #include // 回调函数a void ...
- 输入输出数组元素的函数重载_C ++函数重载| 查找输出程序| 套装3
输入输出数组元素的函数重载 Program 1: 程序1: #include <iostream> using namespace std; class Test { public: vo ...
- c++ swap函数头文件_C++函数模板(泛型编程)
模板是泛型编程的基础,泛型编程即以一种独立于任何特定类型的方式编写代码. 模板是创建泛型类或函数的蓝图或公式.库容器,比如迭代器和算法,都是泛型编程的例子,它们都使用了模板的概念. 每个容器都有一个单 ...
- C++ 泛型编程(一):模板基础:函数模板,类模板,模板原理,模板匹配规则
类模板 函数模板 泛型编程 泛型编程,泛型即是指具有在多种数据类型上皆可操作的含义,其实就是能够帮助开发者编写完全一般化并可重复使用的算法,同样的工作不需要做多次,同样的算法针对不同的类型也不应该写多 ...
- C++ 泛型编程(一):模板基础:函数模板、类模板、模板推演成函数的机制、模板实例化、模板匹配规则
文章目录 泛型编程 函数模板 函数模板实例化 隐式实例化 显式实例化 函数模板的匹配规则 类模板 类模板的实例化 泛型编程 泛型编程旨在削减重复工作,如: 将一个函数多次重载不如将他写成泛型. voi ...
- 泛型编程之函数模板和类模板
1. 函数模板 C++一种编程思想称为泛型编程,主要利用的技术就是模板 C++提供两种模板机制:函数模板和类模板.这里介绍函数模板,类模板在该专题下的另外篇文章中. 函数模板作用:建立一个通用函数,其 ...
- 判断exception类型_C++核心准则T.44:使用函数模板推断类模板参数类型(如果可能)...
T.44: Use function templates to deduce class template argument types (where feasible) T.44:使用函数模板推断类 ...
- binarytreenode”使用 类 模板 需要 模板 参数列表_c++1117 模板核心知识(一)—— 函数模板...
目录 定义函数模板 使用函数模板 两阶段翻译 Two-Phase Translation 模板的编译和链接问题 多模板参数 引入额外模板参数作为返回值类型 让编译器自己找出返回值类型 将返回值声明为两 ...
- c++ 函数模板_C/C++编程笔记:C++入门知识,深入解析C++函数和函数模板
本篇要学习的内容和知识结构概览 函数的参数及其传递方式 1. 函数参数传递方式 传值: 传变量值: 将实参内存中的内容拷贝一份给形参, 两者是不同的两块内存 传地址值: 将实参所对应的内存空间的地址值 ...
最新文章
- Android app开发捷径,让你少去踩坑
- python【蓝桥杯vip练习题库】ALGO-142 P1103(复数运算)
- 硬盘和显卡的访问与控制(一)——《x86汇编语言:从实模式到保护模式》读书笔记01
- 用python内置函数算复杂度吗_番外篇: Python 面试感受
- JS疑惑-2(作用域)
- npm ERR! code EINTEGRITY npm ERR! sha1- 报错解决办法
- Liunx安装gogs,mysql,jdk,tomcat等常用软件
- wifi两种工作模式
- 如何关闭dell inspiron n4010的内置麦克
- 博客园电子期刊2010年2月刊发布啦
- web前端期末大作业实例 (1500套) 集合
- Lock接口和AQS原理与实现(Java并发编程的艺术整理)
- 电气与计算机哪个好考吗,电气工程师难考吗 含金量怎么样
- 银联云闪付控件支付--python版本签名、验签代码
- PUN☀️实用API
- 【ubuntu】解决 Certificate verification failed: The certificate is NOT trusted
- python ljust 中文_python ljust 中文_Python为文档批量注音(生僻字歌词为例)
- 2016校招offer
- 用Qt写的平均绩点计算器
- kindeditor配置syntaxhighlighter...
热门文章
- [靠谱原创!] SSH免密登录设置----原理详解+具体操作(全国人民看完都懂了!)
- 安卓Service完全解析(中)
- 搭建持续集成接口测试平台(Jenkins+Ant+Jmeter)
- php随笔10-thinkphp 3.1.3 模板继承 布局
- JavaScript Array对象 知识点总结
- 《Image Effects 》第一章学习笔记(2)
- python 蓝牙开发_基于python实现蓝牙通信代码实例
- DocumentHelper用法
- Mock(模拟后端接口数据)配合Vuex的使用
- 【oracle11g,17】存储结构: 段的类型,数据块(行连接、行迁移,块头),段的管理方式,高水位线...