模板与泛型编程

函数模板

显示实例化

区别定义与声明

T是模板形参 int是模板实参 

inpunt是函数形参 3是函数实参

显示实例化

模板必须实例化可见 翻译单元一处定义原则

与内联函数异同

引入原因:函数模板是为了编译器两个阶段的处理 内联函数是为了能在编译期展开

模板实参的类型推导

推导原则

推导规则示例

1.

  • 函数形参是左值引用/指针:
    • 忽略表达式类型中的引用
    • 将表达式类型与函数形参模式匹配以确定模板实参

2.万能引用:函数模板的 T&&是万能引用 

  • 如果实参表达式是右值,那么模板形参被推导为去掉引用的基本类型
  • 如果实参表达式是左值,那么模板形参被推导为左值引用,触发引用折叠

引用折叠: 当有两个引用相互绑定时,它们会被折叠成一个引用。在模板实例化期间,引用折叠规则被应用于模板参数。

3、

  • 函数形参不包含引用
    • 忽略表达式类型中的引用
    • 忽略顶层const
    • 数组、函数转换成相应的指针类型

无法推导的情况

  • 模板实参并非总是能够推导得到
    • 如果模板形参与函数形参无关,则无法推导
    • 即使相关,也不一定能进行推导,推导成功也可能存在因歧义而无法使用

即使相关,也不一定能进行推导,推导成功也可能存在因歧义而无法使用

  • 在无法推导时,编译器会选择使用缺省模板实参
    • 可以为任意位置的模板形参指定缺省模板实参——注意与函数缺省实参的区别

但是缺省只能处理无法推导的情况 不能处理能推导但是有歧义的情况

模板缺省实参和函数缺省实参区别

函数缺省实参要从右向左设置。 模板可以不是

自动推导遇到的几种情况

1.SPFINAE

函数模板中的替换失败(Substitution Failure Is Not An Error,简称 SFINAE)是一种编译器处理模板实例化失败的机制。当尝试实例化一个模板时,如果由于某些原因导致实例化失败,编译器并不会报错,而是会尝试使用备选的模板或者进行其他处理。

报错:没有匹配的函数 、忽视函数

重载函数 匹配失败就会找其他模板实例化 忽略掉前一个

2.模板与非模板同时匹配,匹配等级相同,此时选择非模板的版本

下面的代码调用了非模板的fun函数

注意 是匹配等级相同的情况才会选择非模板 如果不同 则会选择更加完美的匹配

下面的代码会调用模板fun

3.多个模板同时匹配,此时采用偏序关系确定选择最特殊的版本

float更加特殊 所以匹配第二个fun

如果同样特殊就会报错

标准类型转换模板

尾置返回类型与类型转换

  • 显式实例化定义:template void fun(int) / template void fun(int)

  • 显式实例化声明:extern template void fun(int) / extern template void fun(int)

  • 注意一处定义原则
  • 来源:stackoverflower

  • 注意实例化过程中的模板形参推导

模板特化

函数模板的(完全)特化:

template<> void f(int) / template<> void f(int)

  • 并不引入新的(同名)名称,只是为某个模板针对特定模板实参提供优化算法

避免使用模板的特化

  • 不参与重载解析,会产生反直觉的效果
  • 通常可以用重载代替
  • 一些不便于重载的情况:无法建立模板形参与函数形参的关联

1.if constexpr解决

 

2.假函数解决

函数模板不能偏特化

C++20auto定义模板参数

类模板

成员函数只有在调用时才会被实例化

证明:程序无法编译 但是去掉21行就能编译

  • 类内类模板名称的简写

类模板成员函数类外定义

成员函数模板

类的成员函数模板

类模板的成员函数模板

类内定义

类外定义

友元函数模板

fun是个友元函数模板

类模板的静态成员

成员对象只有类实例化的时候才会被实例化

static函数只有使用的时候才会实例化

C++11模板参数为友元

类模板的实例化

类模板的显示实例化

类模板特化

完全特化

偏特化

C++17类模板的实参推导

B x(3)自动推导成B<int>

pair的自动推导

C++17之前的解决方法 函数模板推导

C++20新概念

  • 模板的问题:没有对模板参数引入相应的限制
    • 参数是否可以正常工作,通常需要阅读代码进行理解
    • 编译报错友好性较差(vector<int&>)

  • (C++20)Concepts:编译期谓词,基于给定的输入,返回truefalse
    • constraintsrequire从句)一起使用时限制模板参数
    • 通常置于表示模板形参的尖括号后面进行限制

此处限制T是int和float

优点:报错一目了然

Concept的定义与使用

  • 1.包含一个模板参数的Concept
    • 使用requires从句
    • 直接替换typename

  • 2.包含多个模板参数Concept
    • 用做类型constraint时,少传递一个参数,推导出的类型将作为首个参数

requires 表达式 (C++20 起) - cppreference.com   参考资料

  • 简单表达式:表明可以接收的操作

  • 类型表达式:表明是一个有效的类型

  • requires从句所引入的限定具有偏序特性,系统会选择限制最严格的版本

如下 C1更严格 编译器选择匹配C1

  • 特化小技巧:在声明中引入“A||B”进行限制,之后分别针对AB引入特化

重载与模板

数值模板参数与模板模板参数

模板可以接受编译器常量为模板参数

C++17auto value

·

C++20支持浮点数作为模板参数(clang 12不支持)

接受模板作为模板参数

C++17模板的模板考虑缺省实参

clang12支持有限

别名模板

  • 为目标本身引入别名

  • 为类模板的成员引入别名

  • 别名模板不支持特化,但可以基于类模板的特化引入别名,以实现类似特化的功能
    • 注意与实参推导的关系

变长模板

变长模板(Variadic Template)

  • 变长模板参数与参数包

形参包(parameter pack)是C++中用于处理可变数量参数的一种特性。形参包可以接受任意数量的模板参数,并在模板中进行处理。

形参包的基本语法是使用...来表示,可以用在函数模板、类模板以及别的模板上下文中。形参包的展开可以通过递归、折叠表达式(C++17引入)等方式进行。

接数值

接类型

带可选名字的函数形参包

完美转发

右值引用失效

使用万能引用 T&&情况

还是失效 原因:右值引用是个左值

解决方法:转发 forward

  • (C++11)完美转发:std::forward函数
    • 通常与万能引用结合使用
    • 同时处理传入参数是左值或右值的情形

包展开与折叠表达式

消除歧义

internal*p被编译器解读成乘法

  • 使用typenametemplate消除歧义
    • 使用typename表示一个依赖名称是类型而非静态数据成员

使用template表示一个依赖名称是模板

T::internal<A 会被编译器视为小于号比较大小

使用template表示一个依赖名称是模板

template与成员函数模板调用

internal被解读为依赖obj <被解读成小于号

解决方法

C++14变量模板

  • (C++14)变量模板
    • template<typename T> T pi = (T)3.1415926;
    • 其他形式的变量模板

C++14 引入了变量模板(Variable Templates)的概念,它允许你定义参数化的变量,类似于函数模板允许你定义参数化的函数。变量模板提供了一种通用的方式来定义与类型相关的常量或变量,使得代码更具有通用性和灵活性。

lambda模板表达式

// 使用 C++20 中的 lambda 模板auto genericLambda = []<typename T>(T x, T y) {return x + y;};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/209322.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Ignis - Interactive Fire System

Ignis - 点火、蔓延、熄灭、定制! 全方位火焰系统。 这个插件在21年的项目中使用过很好用值使用概述 想玩火吗?如果想的话,那么Ignis就是你的最佳工具。有了Ignis,你可以把任何物体、植被或带皮带骨的网状物转换为可燃物体,它就会自动着火。然后,火焰可以蔓延,点燃其他物…

【docker 】centOS 安装docker

官网 docker官网 github源码 卸载旧版本 sudo yum remove docker \docker-client \docker-client-latest \docker-common \docker-latest \docker-latest-logrotate \docker-logrotate \docker-engine 安装软件包 yum install -y yum-utils \device-mapper-persistent-data…

【优选算法系列】【专题二滑动窗口】第四节.30. 串联所有单词的子串和76. 最小覆盖子串

文章目录 前言一、串联所有单词的子串 1.1 题目描述 1.2 题目解析 1.2.1 算法原理 1.2.2 代码编写 1.2.3 题目总结二、最小覆盖子串 2.1 题目描述 2.2 题目解析 2.2.1 算法原理 2.2.2 代码编写 …

浅谈5G基站节能及数字化管理解决方案的设计与应用-安科瑞 蒋静

截至2023年10月&#xff0c;我国5G基站总数达321.5万个&#xff0c;占全国通信基站总数的28.1%。然而&#xff0c;随着5G基站数量的快速增长&#xff0c;基站的能耗问题也逐渐日益凸显&#xff0c;基站的用电给运营商带来了巨大的电费开支压力&#xff0c;降低5G基站的能耗成为…

actitivi自定义属性(二)

声明&#xff1a;此处activiti版本为6.0 此文章介绍后端自定义属性解析&#xff0c;前端添加自定义属性方法连接&#xff1a;activiti自定义属性&#xff08;一&#xff09;_ruoyi activiti自定义标题-CSDN博客 1、涉及到的类如下&#xff1a; 简介&#xff1a;DefaultXmlPar…

在 JavaScript 中导入和导出 Excel XLSX 文件:SpreadJS

在 JavaScript 中导入和导出 Excel XLSX 文件 2023 年 12 月 5 日 使用 MESCIUS 的 SpreadJS 将完整的 JavaScript 电子表格添加到您的企业应用程序中。 SpreadJS 是一个完整的企业 JavaScript 电子表格解决方案&#xff0c;用于创建财务报告和仪表板、预算和预测模型、科学、工…

图的搜索(一):广度优先搜索算法和深度优先搜索算法

图的搜索&#xff08;一&#xff09;&#xff1a;广度优先搜索算法和深度优先搜索算法 本章主要记录了图的搜索算法&#xff0c;和可以解决图的基本问题——最短路径问题的算法。本章主要对图搜索的相关算法进行了介绍&#xff1a;广度优先搜索算法、深度优先搜索算法。 下一…

公网域名如何解析到内网IP服务器——快解析域名映射外网访问

在本地搭建主机应用后&#xff0c;由于没有公网IP或没有公网路由权限&#xff0c;在需要发布互联网时&#xff0c;就需要用到外网访问内网的一些方案。由于内网IP在外网不能直接访问&#xff0c;通常就用通过外网域名来访问内网的方法。那么&#xff0c;公网域名如何解析到内网…

权威认证!景联文科技入选杭州市2023年第二批省级“专精特新”中小企业认定名单

为深入贯彻党中央国务院和省委省政府培育专精特新的决策部署&#xff0c;10月7日&#xff0c;杭州市经济和信息化委员会公示了2023年杭州“专精特新”企业名单&#xff08;第二批&#xff09;。 根据工业和信息化部《优质中小企业梯度培育管理暂行办法》&#xff08;工信部企业…

【Vue3+Ts项目】硅谷甄选 — 路由配置+登录模块+layout组件+路由鉴权

一、路由配置 项目一共需要4个一级路由&#xff1a;登录&#xff08;login&#xff09;、主页&#xff08;home&#xff09;、404、任意路由&#xff08;重定向到404&#xff09;。 1.1 安装路由插件 pnpm install vue-router 1.2 创建路由组件 在src目录下新建views文件…

Graphpad Prism10.1.0 安装教程 (含Win/Mac版)

GraphPad Prism GraphPad Prism是一款非常专业强大的科研医学生物数据处理绘图软件&#xff0c;它可以将科学图形、综合曲线拟合&#xff08;非线性回归&#xff09;、可理解的统计数据、数据组织结合在一起&#xff0c;除了最基本的数据统计分析外&#xff0c;还能自动生成统…

Python:核心知识点整理大全8-笔记

目录 ​编辑 4.5 元组 4.5.1 定义元组 dimensions.py 4.5.2 遍历元组中的所有值 4.5.3 修改元组变量 4.6 设置代码格式 4.6.1 格式设置指南 4.6.2 缩进 4.6.3 行长 4.6.4 空行 4.6.5 其他格式设置指南 4.7 小结 第5章 if语句 5.1 一个简单示例 cars.py 5.2 条…

现代皮质沙发模型材质编辑

在线工具推荐&#xff1a; 3D数字孪生场景编辑器 - GLTF/GLB材质纹理编辑器 - 3D模型在线转换 - Three.js AI自动纹理开发包 - YOLO 虚幻合成数据生成器 - 三维模型预览图生成器 - 3D模型语义搜索引擎 当谈到游戏角色的3D模型风格时&#xff0c;有几种不同的风格&#xf…

线性容器(QByteArray、QString、QList模板类)、堆栈窗体

QT 线性容器 点击查看&#xff1a;字符和字节的区别&#xff0c;ASCII、Unicode 和 UTF-8 编码的区别。&#xff08;&#x1f448; 安全链接&#xff0c;放心跳转&#xff09; QByteArray 思考&#xff1a;char buf[6] “hello”; 如果 C 语言中要利用 buf 内容重新生成 “…

学生备考使用台灯到底好不好?公认好用的护眼台灯推荐

在现代生活中&#xff0c;许多学生的学习压力越来越大&#xff0c;面临的近视几率也越来越大&#xff0c;特别是初中生&#xff0c;眼睛发育还未完全&#xff0c;使用不恰当的灯光也会对眼睛造成损害&#xff0c;特别是护眼台灯。虽然护眼台灯在功能上能够提供充足、柔和的光线…

《文存阅刊》期刊发表简介

《文存阅刊》以“深研文化创新&#xff0c;崇尚科学真理&#xff0c;坚持双百方针&#xff0c;打造学术精品”为办刊宗旨&#xff0c;涵盖艺术、文学、社科等多项内容&#xff0c;适应了文化市场需求&#xff0c;很好的回应了广大文化理论工作者的关切&#xff0c;为下一步打造…

ChatGPT新媒体运营神器:轻松驾驭内容创作与传播

文章目录 1. 内容创作2. 社交媒体管理3. 用户互动与客户服务 《巧用ChatGPT轻松玩转新媒体运营》内容简介作者简介目录前言/序言本书内容本书特色本书读者对象获取方式 随着互联网的高速发展&#xff0c;新媒体已经成为了人们获取信息、交流思想的重要渠道。在这个信息爆炸的时…

【SpringCache】快速入门 通俗易懂

1. 介绍 Spring Cache 是一个框架&#xff0c;实现了基于注解的缓存功能&#xff0c;只需要简单地加一个注解&#xff0c;就能实现缓存功能。 Spring Cache 提供了一层抽象&#xff0c;底层可以切换不同的缓存实现&#xff0c;例如&#xff1a; EHCache Caffeine Redis(常用…

Centos7、Mysql8.0 load_file函数返回为空的终极解决方法--暨selinux的深入理解

零、问题背景 最近想换房&#xff0c;为了方便自己对比感兴趣的房子&#xff0c;因此决定将目标房源的基本信息放在表里&#xff0c;特别是要一目了然的看到众多房子的各种图纸和照片&#xff0c;因此决定要在Mysql8.0.34数据库中以二进制形式保存图片&#xff08;抛开合理性和…

Vue 2.0源码分析-update

Vue 的 _update 是实例的一个私有方法&#xff0c;它被调用的时机有 2 个&#xff0c;一个是首次渲染&#xff0c;一个是数据更新的时候&#xff1b;由于我们这一章节只分析首次渲染部分&#xff0c;数据更新部分会在之后分析响应式原理的时候涉及。_update 方法的作用是把 VNo…