跟我学c++中级篇——模板的基础术语说明
一、类模板术语
1、模板的特化
模板的特化也叫具体化,非常容易理解,就是把模板中的模板参数给定具体的类型。看下面的例子:
//模板
template <typename T,typname N>
class Data
{};
//特化
template<>
class Data<int,int>
{};
特化,特,就是特别的,也就是不同于广泛的,泛型中的特例,这就好明白了。记住,特化前必须有泛型化的定义。
2、全特化
特化分为全特化和偏特化,全特化(full specialization)又称显示特化(explicit specialization)或者显示具体化。指的是把模板类型参数全部给定,像上面的例子就是,这里再举一个简单的:
//模板
template <typename T>
class Data
{};
//全特特化
template<>
class Data<int>
{};
注意,全特化可以有多个版本,如上面的可以是int,也可以是short,也可以是等等。
3、偏特化
偏特化(Partial Specialization),也称为局部特化或者部分特化,顾名思义,其实就是只对部分模板参数进行了特化,也即指定类型。这里不举比较容易理解的,举几个不常见的,一般指针也被当成偏特化:
//模板
template <typename T>
class Data
{};
//指针偏特化
template <typename T>
class Data<T*>
{};
///
//模板
template <typename T,typname N>
class Data
{};
//相同参数偏特化
template<typename T>
class Data<T,T>
{};
//单独偏特化
template<typename N>
class Data<int,N>
{};
4、实例化和显示实例化
实例化也好理解,就是具体使用模板类或者模板函数,就是实例化。但是这里有一个特别的,叫做显示实例化,Explicit Instantiation,它也叫强制实例化,也叫显示具现化(有的可能也叫全实例化,但这个不太清楚,没找到出处)。来看一下它的代码:
template <typename T>
class Stack{};
template class Stack<int>;
和特化的区别在形式上特化有template<>而显示实例化只有template。全特化是指模板参数全部指定,算是活化中的一个特例。而显示实例化是真正的定义出来这个函数,用模板的语义但却如普通类和函数一样使用了。也就是说,实例化的模板,就不再是模板了,和普通的类和函数没有差别。
举一个例子,生产一种配件,正常情况是一种模具(泛型模板),当材料是铜质时,用同一个模板,但里面加了一个标记(特化),而当真正的材料压进模具,出来成品叫实例化。
显示实例化只能一次,而且要在CPP文件中。
4、变量模板
这个在前面刚刚分析过,和constexpr在一起可以在元编程里起到很重要的作用。看下面的代码:
template<typename T>
constexpr T pi{3.1415926535897932385};
注意,这玩意儿也全特化和偏特化,而且最可有意思的是,它的特化类型并不要求与特化模板匹配。
5、非类型模板参数和模板默认参数
这个也很好理解,那模板的形参不是一个不确定的参数而一个已知的类型,默认参数就更好理解了,将其值定义一个确定的值,如果在实例化模板过程中没有给其定义实参,就使用默认的类型为实参,看代码:
//函数非类型模板参数
template<int Val, typename T>T addValue (T x){return x + Val;
}
//类非类型模板参数
template<int const length>
class Buffer
{...};
//默认参数
template<typename T, typename Cont = std::vector<T>>
class Stack {...};
//函数模板默认参数
template<typename T>
void test (T t, int number = 10);
template <typename T,typename N = int>
void test(T t,N n = 0){...}
6、SFINAE和Concepts
Substitution Failure Is Not An Error,替换失败不是错误。用来做模板编译期检查的,常用于模板元编程。可以说如果比较常用到这玩意儿,开发者的水平应该是比较高了。Concepts,概念,可以理解成它的升级版,这个怎么用,在前面反复分析过,这里不再赘述。
7、外部模板
Extern Template,C++11的新功能,主要是减轻编译在大规模编译模板时产生的重复类型实例,降低编译器的工作。它依赖于前面提到的显示实例化,看下面的代码:
//t.h
template <typename T>
void test(T) { }
// t1.cpp
#include "t.h"
template void test<int>(int); // 显式实例化
void call()
{test(0);
}// t2.cpp
#include "t.h"
extern template void fun<int>(int); // 外部模板的声明
void invoke()
{test(3);
}
8、变参模板
c++11推出的参数不固定的模板,看下面的代码:
template <typename ...Args>
class Ex{....};
二、函数模板术语
函数模板与上面术语基本类似,但是函数模板没有偏特化,试想一下,偏特化和重载怎么容易区别呢?但是,在《c++ template》第二版中,却对此提出了一些看法,并且这些想法在c++标准委员会上进行过讨论,只不过没有引起重视罢了。有兴趣可以看看,至于未来会是如何,没人敢说会怎么样。下面看相关的例子:
1、全特化
template <typename T>
void test(T) {...}template<> void test<int>(int); //全特化
2、显示实例化
函数的显示实例化:
template <typename T>
void test(T) {...}//cpp
template void test<int>(int); // 显式实例化
3、变参模板
template <typename ...Args>
void test(Args ... args)
{...}
三、模板中的一些具体术语应用
1、this
在普通的类函数中,一般是推荐使用this用来让代码更清晰的表明这是一个类对象的成员。但好多开发者为了简单,就不写了。但是在模板类的继承中,如果子类继承了父类的成员,如果不显示的使用this,会导致编译错误。这时只需要写上this即可。
2、template限定符
这个前面分析过,就是为了让编译器知道使用的是模板类型,帮助编译器进行类型推导的,类似于:
typename Shell<T>::template In<N>::template Deep<N>& p) {
p.template Deep<N>::f(); // inhibit virtual call
}
3、模板的模板参数
也叫双重模板参数,这个在前面也提到过,就是模板的参数,仍然是模板,类似于下面:
template <template <typename T> class N>
class A{...};
//C++17后
template <template <typename T> typename N>
class A{...};
注意,新手不要把class单纯当成一个类关键字,所以想当然的使用struct,union这些关键字,使用后者都是错误的。切记。
4、右尖括号的优化
在c++11以前,如果模板实例化中遇到连续的两个>>符号,编译器会把它当成右移符号,从而引起编译错误,所以要在两个右尖括号中间打一个空格,这也是看老的代码时的一个风景,但在c++11之后已经优化正常使用了。例如:
template <typename T>
class Data
{typedef T type;};
template <typename T>
class D
{};Data<D<int>>::type d;//此御的右双尖括号
但是这种方法在处理真的">>"右移时,又会和新编译有冲突,错误和解决方式如下:
template <size_t T>
class D
{};D<100>>2> xx;//C++11 error
D<(100>>2)> xx;//OK
5、模板的别名
这个也是为了简化模板编程使用的,都知道在模板编程中,可能会有一个很长的类型定义,但typedef又无法在这种情况下作用,所以老的标准中,一般是定义一个模板类,再把此类型定义到这个模板内进行变相的使用,增加了很多的工作量和理解的复杂度。类似于下面:
typedef unsigned int uint32 //ok
//但是要类似偏特化某个类型,比如下面固定KEY为std::string ,value类型可变
typedef std::map<std::string,std::string> map_string
typedef std::map<std::string,int> map_string
//C++11 prev
template <typename T>
class map_string{typedef std::map<std::string ,T> type;
}
map_string<std::string>::type m1;
//C++11
template <typename T>
using map_string = std::map<std::string,T>;
map_string m1;
6、外部定义成员模板
// member templates outside class
template<typename T>
class A
{
public:template<typename S>void test(const S &s);
};template<typename T>
template <typename S>
void A<T>::test(const S &s)
{...}
四、总结
这些术语有些是比较为通用的,有的是比较少用的,但在某些书籍中又比较权威,所以了解一些还是有用的。特别是侯捷老师的书里,用法有些不同大家可以对比来印证就知道了什么意思。
这里面其实每个都可以单独拿出来讲很多,本篇主要是综合一下,做一个纲领性的说明。以后如果标准继续更迭,可能会继续在此基础上不断的完善。
相关文章:
跟我学c++中级篇——模板的基础术语说明
一、类模板术语 1、模板的特化 模板的特化也叫具体化,非常容易理解,就是把模板中的模板参数给定具体的类型。看下面的例子: //模板 template <typename T,typname N> class Data {}; //特化 template<> class Data<int,int&…...
最新Win10离线安装.NET Framework 3.5的方法(附离线包2022/3/22)
win10系统安装软件时,可能需要.net framework3.5的运行环境,当我们安装某些软件的时候会提示“你的电脑上的应用需要使用以下Windows功能:.NET Framework 3.5(包括.NET 2.0和3.0)。如果系统默认的是4.0以上的版本,当软件需要.net framework3.…...
最新docker多系统安装技术
在Ubuntu操作系统中安装Docker 在Ubuntu操作系统中安装Docker的步骤如下。 1.卸载旧版本Docker 卸载旧版本Docker的命令如下: $ sudo apt-get remove docker docker-engine docker.io 2.使用脚本自动安装 在测试或开发环境中࿰…...
系统架构设计高级技能 · 云原生架构设计理论与实践
系列文章目录 系统架构设计高级技能 软件架构概念、架构风格、ABSD、架构复用、DSSA(一)【系统架构设计师】 系统架构设计高级技能 系统质量属性与架构评估(二)【系统架构设计师】 系统架构设计高级技能 软件可靠性分析与设计…...
Springboot集成RocketMQ——简单使用
目录 1.MQ选型 2.RocketMQ基本架构 3.Springboot集成RocketMQ 4.顺序消息 5.延时消息 6.事务消息 1.MQ选型 目前市面上的MQ选型:主要分为3个类型 Kafka:吞吐量大,且性能好,集群高可用;会丢失数据,功…...
第一百二十四回 Flexible组件
文章目录 概念介绍使用方法示例代码 我们在上一章回中介绍了扩展内容相关的知识,本章回中将介绍 Flexible组件.闲话休提,让我们一起Talk Flutter吧。 概念介绍 我们在前面章回中介绍了扩展列表相关的内容,当页面中其它组件和扩展列表一起使…...
关于stm32推挽带有上下拉电阻的思考、IO口驱动能力是什么
1、发现推挽带有上下拉电阻 1.1、stm32手册 记忆中推挽是不需要上下拉的,没关注过,但是我真的理解上下拉吗,下图来自stm32f4的中文版和英文版的数据手册,没有翻译错,就是“推挽带有上下拉的能力”。 1.2、查找相关信…...
考研408 | 【操作系统】 内存管理
内存的基础 内存和内存的作用: 几个常用的数量单位: 指令的工作原理: 问题:如何将指令中的逻辑地址转换为物理地址? 解决办法:装入的三种方式 1.绝对装入 2.可重定位装入 3.动态重定位 从写程序到程…...
C# 工厂模式
一、概述 工厂模式(Factory Pattern)是一种创建型设计模式,它提供了一种创建对象的最佳方式。在C#中,工厂模式通过定义一个公共接口或抽象类来创建对象,而具体的对象创建则由工厂类来实现。 工厂模式主要包含三个角色…...
在云服务器上安装Jenkins
说明:Jenkins是一个部署项目的平台,通过Jenkins可以省去从项目开发–>部署项目之间的所有流程,做到代码提交即上线。本文介绍在云服务CentOS上安装Jenkins。 前提 安装Jenkins之前,先要在云服务上安装JDK、Maven、Git&#x…...
一文了解SpringBoot中的IOC
目录 1.什么是IOC 2.IOC容器 3.创建IOC容器 4.装配Bean到IOC容器 5.依赖注入 1.什么是IOC IOC:Inversion of Control 控制反转 Sping中我们把一个个对象称为Bean,以前我们实例一个对象的时候,都会直接New一个 而在Spring中࿰…...
docker-compose管理创建LNMP服务并运行Wordpress网站平台
文章目录 一.项目环境1. 环境描述2.项目需求 二.部署过程1.安装Docker2.安装Docker加速器3.Docker-Compose安装部署4.准备依赖文件、配置nginx5.配置mysql6.配置php7.编写docker-compose.yml8.验证 三.容器快照,然后将Docker镜像打包成tar包备…...
【宝藏系列】一文带你梳理 Linux 的五种 IO 模型
【宝藏系列】一文带你梳理 Linux 的五种 IO 模型 文章目录 【宝藏系列】一文带你梳理 Linux 的五种 IO 模型👨🏫前言1️⃣用户态和核心态1️⃣1️⃣用户态和核心态的切换 2️⃣进程切换3️⃣进程阻塞4️⃣文件描述符(fd, File Descriptor)5️⃣缓存I/O…...
【Python】模块、包
模块 Python模块(Module),是一个Python文件,以.py结尾。模块能定义函数,类和变量,模块里也能保护可执行的代码。 不同模块,同名的功能,如果都被导入,那么后者会覆盖前者…...
CMAKE_CUDA_ARCHITECTURES针对Jetson Xavier或者Orin的设置
不同jetson设备对应不同的CMAKE_CUDA_ARCHITECTURES的设置,如下: # TX1, Nano ------ 53 # TX2 ------ 62 # AGX Xavier, NX Xavier ------ 72 # AGX Orin, NX Orin ----…...
sqlite3.OperationalError: unable to open database file解决方法
执行superset时,提示该错误:sqlite3.OperationalError: unable to open database file 由于superset里使用django设置sqlite3数据库。 应该属于django设置sqlite3数据库的问题: OperationalError: unable to open database file 原因 1&a…...
SSL核心概念 SSL类型级别
SSL:SSL(Secure Sockets Layer)即安全套接层,及其继任者传输层安全(Transport Layer Security,TLS)是为网络通信提供安全及数据完整性的一种安全协议。TLS与SSL在传输层对网络连接进行加密。 H…...
器件介绍TMP1826NGRR、TMP1826DGKR、TMP1827NGRR、TMP1075NDRLR数字温度传感器
一、TMP1826 具有 2Kb EEPROM 的 1-Wire、0.2C 精度温度传感器 器件介绍 TMP1826 是一款高精度、1-Wire 兼容的数字输出温度传感器,具有集成的 2Kb EEPROM 和 –55C 至150C 的宽工作温度范围。TMP1826 在 10C 至45C 的温度范围内提供 0.1C(典型值&#…...
抖店必须绑定抖音账号吗?聊6个抖店不为人知的小细节,别外传
我是王路飞。 现在做抖店,比如绑定一个抖音账号吗? 了解过抖店的朋友都知道,之前开通抖音小店,是需要绑定一个抖音号作为店铺的官方账号的。 而且属于硬性规定,必须要绑定,否则店铺无法正常运营。 但是…...
如何搭建智能家居系统并通过内网穿透实现远程控制家中设备
文章目录 前言1. 安装Home Assistant2. 配置Home Assistant3. 安装cpolar内网穿透3.1 windows系统3.2 Linux系统3.3 macOS系统 4. 映射Home Assistant端口5. 公网访问Home Assistant6. 固定公网地址6.1 保留一个固定二级子域名6.2 配置固定二级子域名 前言 Home Assistant&…...
uni-app学习笔记二十二---使用vite.config.js全局导入常用依赖
在前面的练习中,每个页面需要使用ref,onShow等生命周期钩子函数时都需要像下面这样导入 import {onMounted, ref} from "vue" 如果不想每个页面都导入,需要使用node.js命令npm安装unplugin-auto-import npm install unplugin-au…...
Leetcode 3577. Count the Number of Computer Unlocking Permutations
Leetcode 3577. Count the Number of Computer Unlocking Permutations 1. 解题思路2. 代码实现 题目链接:3577. Count the Number of Computer Unlocking Permutations 1. 解题思路 这一题其实就是一个脑筋急转弯,要想要能够将所有的电脑解锁&#x…...
Spring Boot面试题精选汇总
🤟致敬读者 🟩感谢阅读🟦笑口常开🟪生日快乐⬛早点睡觉 📘博主相关 🟧博主信息🟨博客首页🟫专栏推荐🟥活动信息 文章目录 Spring Boot面试题精选汇总⚙️ **一、核心概…...
多模态大语言模型arxiv论文略读(108)
CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文标题:CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文作者:Sayna Ebrahimi, Sercan O. Arik, Tejas Nama, Tomas Pfister ➡️ 研究机构: Google Cloud AI Re…...
Java面试专项一-准备篇
一、企业简历筛选规则 一般企业的简历筛选流程:首先由HR先筛选一部分简历后,在将简历给到对应的项目负责人后再进行下一步的操作。 HR如何筛选简历 例如:Boss直聘(招聘方平台) 直接按照条件进行筛选 例如:…...
Spring AI与Spring Modulith核心技术解析
Spring AI核心架构解析 Spring AI(https://spring.io/projects/spring-ai)作为Spring生态中的AI集成框架,其核心设计理念是通过模块化架构降低AI应用的开发复杂度。与Python生态中的LangChain/LlamaIndex等工具类似,但特别为多语…...
AI书签管理工具开发全记录(十九):嵌入资源处理
1.前言 📝 在上一篇文章中,我们完成了书签的导入导出功能。本篇文章我们研究如何处理嵌入资源,方便后续将资源打包到一个可执行文件中。 2.embed介绍 🎯 Go 1.16 引入了革命性的 embed 包,彻底改变了静态资源管理的…...
SiFli 52把Imagie图片,Font字体资源放在指定位置,编译成指定img.bin和font.bin的问题
分区配置 (ptab.json) img 属性介绍: img 属性指定分区存放的 image 名称,指定的 image 名称必须是当前工程生成的 binary 。 如果 binary 有多个文件,则以 proj_name:binary_name 格式指定文件名, proj_name 为工程 名&…...
NXP S32K146 T-Box 携手 SD NAND(贴片式TF卡):驱动汽车智能革新的黄金组合
在汽车智能化的汹涌浪潮中,车辆不再仅仅是传统的交通工具,而是逐步演变为高度智能的移动终端。这一转变的核心支撑,来自于车内关键技术的深度融合与协同创新。车载远程信息处理盒(T-Box)方案:NXP S32K146 与…...
Java求职者面试指南:计算机基础与源码原理深度解析
Java求职者面试指南:计算机基础与源码原理深度解析 第一轮提问:基础概念问题 1. 请解释什么是进程和线程的区别? 面试官:进程是程序的一次执行过程,是系统进行资源分配和调度的基本单位;而线程是进程中的…...
