0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

C++中为什么需要函数模板、类模板?

FPGA之家 来源:嵌入式客栈 作者:嵌入式客栈 2021-03-12 14:31 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

[导读] 最近使用C++做些编程,把日常遇到的些比较重要的概念总结分享一下。本文来分享一下模板类的原理,以及为什么需要模板类,使用时的基本要点。

为什么需要模板

比如需要设计一个描述点的类,大致很快可以写成这样:

class Point_F { public: /*默认传入参数为0,0*/ Point_F(float x0 = 0, float y0 = 0) :x(x0), y(y0) /*初始化列表*/ { } /*用const修饰函数,表示函数不会修改成员数据*/ float get_x() const { return x; } float get_y() const { return y; } private: /*一般会将数据放在私有区,以对外隐藏*/ float x; float y; };

可问题是,在有的场合这点的坐标系有可能不需要浮点,比如界面设计中点往往是整型表示即可,那此时就需要再设计一个整型成员类:

class Point_I { public: /*默认传入参数为0,0*/ Point_f(int x0 = 0, int y0 = 0) :x(x0), y(y0) /*初始化列表*/ { } /*用const修饰函数,表示函数不会修改成员数据*/ int get_x() const { return x; } int get_y() const { return y; } private: /*一般会将数据放在私有区,以对外隐藏*/ int x; int y; };

可是在应用代码中,往往发现对于不同数据成员的应用操作确实基本类似,而且应用代码往往这两种(甚至更多成员数据类型)都可能会同时用到,仅仅因为数据类型就需要笨笨的将原代码在改写一下,在现代高级语言中,这显然就比较机械了。

C++中有没有可能将不同成员数据类型但是其顶层逻辑相同的对象,设计为一个类呢?就比如:

C++模板编程正是为了解决这样的需求而设计的机制。该机制允许函数或类使用泛型类型(generic type)进行操作。从而,函数或类就可以处理许多不同的数据类型,而无需为每种数据类型重写相应的类或者函数。

怎么实现的呢?

这里又可以大致分这样三种情况:

函数模板(Function templates)

类模板(Class templates)

**成员模板(Member templates) **

函数模板

函数模板其基本语法范式为:

template 《class identifier》 function_declaration; template 《typename identifier》 function_declaration;

template 为模板关键字

《typename identifier》 、《class identifier》 定义函数参数泛型类型或函数体类变量泛型类型

比如:

#include 《iostream》 using namespace std; template 《typename T》 T max(T a, T b) { return a 》 b ? a : b ; }

又或者写成如下形式:

#include 《iostream》 using namespace std; template 《class T》 T max(T a, T b) { return a 》 b ? a : b ; }

那么或许有的朋友会任务关键字class就意味着自定义类,而typename则是基本数据类型,比如int,float等,这样理解其实是不对的,从C++编译器的角度template 《typename T》与template 《class T》其语义是一样的,都是泛型,用户在使用这个模板函数的时候,所传入的参数都既可以是基本数据类型,也可以是类名。

对于上面的代码,或许初使用的朋友还会问,是不是可以随便传入类,这有可能编译不过。为什么呢?你传入的类需要支持》操作符,如果对于某个类你想使用该函数,而本身不支持》操作符,则需要实现》操作符。

类模板

与函数模板类似,类内部成员数据或者函数的参数或变量会使用,模板关键字定义的泛型名。比如:

template 《typename T》 class Point_T { public: Point_T(T x0 = 0, T y0 = 0) :x(x0), y(y0) { } T get_x() const { return x; } T get_y() const { return y; } private: T x; T y; };

这小段代码就回答了之前提出的问题,可以支持不同数据类型的点。

int main() { Point_T《int》 p1(1, 2); Point_T《float》 p2(1.1f, 2.2f); cout 《《 p1.get_x() 《《 endl 《《 p1.get_y() 《《 endl; cout 《《 p2.get_x() 《《 endl 《《 p2.get_y() 《《 endl; }

以上述简单例子看,分别构造了整型点p1,以及浮点型点p2,那么究竟怎么做到的呢?为了理解得更清楚,这里将其关键汇编代码段拷贝下来简要看看:

Point_T《int》 p1(1, 2); 000C1D6C push 2 000C1D6E push 1 000C1D70 lea ecx,[p1] 000C1D73 call Point_T《int》::Point_T《int》 (0C11D1h) Point_T《float》 p2(1.1f, 2.2f); 000C1D78 push ecx 000C1D79 movss xmm0,dword ptr [__real@400ccccd (0C7B34h)] 000C1D81 movss dword ptr [esp],xmm0 000C1D86 push ecx 000C1D87 movss xmm0,dword ptr [__real@3f8ccccd (0C7B30h)] 000C1D8F movss dword ptr [esp],xmm0 000C1D94 lea ecx,[p2] 000C1D97 call Point_T《float》::Point_T《float》 (0C1064h)

可见编译器对不同类型参数实际上做了相应解析,相当于根据用户程序传入的参数编译出相应的多份代码。所以可以简单理解成编译器根据不同泛型实际参数类型生成了相应的处理代码。而前面所说的模板函数其原理也基本类似。

总结一下

通过些简单例子,梳理一下模板函数以及模板类的基本概念以及原理,理解了这两个概念,就比较容易理解成员模板。所谓泛型模板编程,其本质是编译器针对不同参数类型解析解析生成相应的处理代码。学会使用模板泛型编程你会发现你会少写很多代码,代码看起来会比较优雅,而其实操作起来也没有想象中那么难。

原文标题:什么是函数模板、类模板?怎么做到的?

文章出处:【微信公众号:FPGA之家】欢迎添加关注!文章转载请注明出处。

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 编程
    +关注

    关注

    90

    文章

    3723

    浏览量

    97430
  • 模板
    +关注

    关注

    0

    文章

    111

    浏览量

    21122
  • C++
    C++
    +关注

    关注

    22

    文章

    2128

    浏览量

    77360

原文标题:什么是函数模板、类模板?怎么做到的?

文章出处:【微信号:zhuyandz,微信公众号:FPGA之家】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    Helm包管理与模板化部署实战

    直接用kubectl管理K8s资源,10个微服务就要维护几十个YAML文件,版本管理靠文件夹命名,回滚靠手动替换文件。Helm把一组相关的K8s资源打包成Chart,支持模板化、版本管理、一键部署和回滚,是K8s生态事实上的包管理标准。
    的头像 发表于 02-26 16:37 433次阅读

    瑞萨RA-Eco-RA4M2开发板基本模板搭建与LED指南

    瑞萨“RA MCU众测宝典”环境搭建专题再添硬核实操!这次将解锁“基本模板搭建与LED”技能,加入简易调度器实现多任务管理,一步步搞定“能直接落地”的开发模板,不管是入门练手还是项目开发都能复用。
    的头像 发表于 02-24 14:43 6738次阅读
    瑞萨RA-Eco-RA4M2开发板基本<b class='flag-5'>模板</b>搭建与LED指南

    keil实现cc++混合编程

    参考touchgfx生成的代码,发现了一个不需要添加--cpp11 参数的解决方法,具体操作如下。 一、创建一个空白的C文件和头文件在头文件定义c++文件
    发表于 01-26 08:58

    京东关键词 API 场景的需求梳理模板

    京东关键词 API 场景需求梳理模板模板适用于 联盟推广、商家运营、工具开发、市场分析 等基于京东关键词 API 的业务需求梳理,可直接用于需求文档撰写、API 对接方案评审。 核心模块 填写项
    的头像 发表于 01-08 13:45 269次阅读

    CW32F030开发板的工程模板

    cw32f030-tft-demo也可以作为你以后移植工程模板。 CW32F030开发板工程模板优化 文件结构如下: Application: main.c main.h
    发表于 12-26 08:08

    C语言与C++的区别及联系

    创建源文件时什么都不给,默认是.cpp。 3、返回值 C语言中,如果一个函数没有指定返回值类型,默认返回int类型;C++,如果一个函数
    发表于 12-24 07:23

    C语言和C++之间的区别是什么

    区别 1、面向对象编程 (OOP): C语言是一种面向过程的语言,它强调的是通过函数将任务分解为一系列步骤进行执行。 C++C语言的基础上扩展了面向对象的特性,支持
    发表于 12-11 06:23

    目标追踪的简易实现:模板匹配

    。 因此,模板匹配算法步骤就很明确了:用模板在整帧图像扫描,对于一个超清视频(1280*720 P)如果我们用 50*50 大小的模板进行匹配,扫描完整个图像就
    发表于 10-28 07:21

    店铺装修模板同步接口技术解析

    ​  在电商平台开发,店铺装修模板的跨系统同步是核心需求。本文从接口设计、数据结构和实现逻辑三个维度进行技术拆解。 一、接口设计规范 基础参数 请求方法:POST 端点路径:/api/v1
    的头像 发表于 10-17 15:24 425次阅读
    店铺装修<b class='flag-5'>模板</b>同步接口技术解析

    Allegro Skill工艺辅助之导入叠层模板

    在PCB设计,导入叠层模板能够确保设计的标准化和规范化,避免因手动设置叠层参数而可能出现的错误或不一致情况。
    的头像 发表于 07-10 17:10 3496次阅读
    Allegro Skill工艺辅助之导入叠层<b class='flag-5'>模板</b>

    HarmonyOS5云服务技术分享--Serverless抽奖模板部署

    startRaffle) 结束抽奖回调函数(比如endRaffle) 记下函数名,后续配置要用到 3️⃣ ​​工程包准备​​ 下载官方提供的抽奖模板工程包 解压后重点检查portal/dist目录结构 记得
    发表于 05-22 20:25

    基于LockAI视觉识别模块:C++模板匹配

    模板匹配是一种在图像同时寻找多个模板的技术。通过对每个模板逐一进行匹配,找到与输入图像最相似的区域,并标记出匹配度最高的结果。本实验提供了一个简单的多
    发表于 05-14 15:00

    基于LockAI视觉识别模块:C++模板匹配

    模板匹配是一种在图像同时寻找多个模板的技术。通过对每个模板逐一进行匹配,找到与输入图像最相似的区域,并标记出匹配度最高的结果。本实验提供了一个简单的多
    的头像 发表于 05-14 14:37 1746次阅读
    基于LockAI视觉识别模块:<b class='flag-5'>C++</b>多<b class='flag-5'>模板</b>匹配

    基于LockAI视觉识别模块:C++模板匹配

    模板匹配是一种在图像寻找特定模式的技术。它通过滑动一个模板图像(较小的图像)在输入图像上进行比较,找到最相似的区域。本实验提供了一个简单的模板匹配案例,并将其封装为一个自定义
    发表于 05-13 14:40

    基于LockAI视觉识别模块:C++模板匹配

    模板匹配是一种在图像寻找特定模式的技术。它通过滑动一个模板图像(较小的图像)在输入图像上进行比较,找到最相似的区域。本实验提供了一个简单的模板匹配案例,并将其封装为一个自定义
    的头像 发表于 05-13 14:14 900次阅读
    基于LockAI视觉识别模块:<b class='flag-5'>C++</b><b class='flag-5'>模板</b>匹配