【C++笔记】模版的特化及其编译分离

【C++笔记】模版的特化及其编译分离

🔥个人主页大白的编程日记

🔥专栏C++笔记


文章目录

  • 【C++笔记】模版的特化及其编译分离
    • 前言
    • 一.模版
      • 1.1非类型模板参数
    • 二.模板的特化
      • 2.1特化的定义
      • 2.2 函数模板特化
      • 2.3底层const
      • 2.4 类模板特化
    • 三.模板分离编译
      • 3.1 什么是分离编译
      • 3.2 模板的分离编译
    • 四.模版总结
    • 后言

前言

哈喽,各位小伙伴大家好!上期我们讲了容器适配器和deque。今天我们来讲一下模版进阶。话不多说,我们进入正题!向大厂冲锋
在这里插入图片描述

一.模版

1.1非类型模板参数

模板参数分类类型形参与非类型形参。

类型形参即:出现在模板参数列表中,跟在class或者typename之类的参数类型名称。

非类型形参,就是用一个常量作为类(函数)模板的一个参数,在类(函数)模板中可将该参数当成常量来使用。

非类型参数是个常量。
非类型参数通常用于定义静态数组或其他容器的大小。

// 定义一个模板类型的静态数组
template<class T, size_t N = 10>
class array
{
private:T _array[N];size_t _size;
};
  • 对比宏
    宏也可以实现类似的效果.
    但是宏的定义是写死的,而我们的非类型参数可以是任意大小。

    例如这里宏定义就只能实现5个大小的容器。

  • 原理
    非类型的模板参数必须在编译期就能确认结果。
    非类型参数的原理还是生成了多个类。
    N是5就是实例化出_array[5]的类

  • 指定整形
    浮点数、类对象以及字符串是不允许作为非类型模板参数的。C++20才支持。
    因为非类型参数数主要是为了方便固定值去定义容器的大小。

  • 缺省值
    非类型参数可以给缺省值。

    那如果没有模版参数且非类型参数有缺省值。我们是不是可以什么都不用传。

但是不支持,C++20之后才支持什么都不传。
C++20之前需要加上<>。

同时非类型参数可以有多个,同时bool也是属于整形家族
所以这里bool也可以做非类型模版参数

namespace qcj
{// 定义一个模板类型的静态数组template<size_t N = 10, bool flag=1>class array{private:int _array[N];size_t _size;};
}
  • array和静态数组

库里的array就是用到了非类型模版参数。
array和静态数组有什么优势呢?
array在越界检查方面更严格。
同时array开空间效率更高。

二.模板的特化

2.1特化的定义

通常情况下,使用模板可以实现一些与类型无关的代码,但对于一些特殊类型的可能会得到一些错误的结果,需要特殊处理,比如:实现了一个专门用来行小于比较的函数模板

在这里插入图片描述

可以看到,Less绝对多数情况下都可以正常比较,但是在特殊场景下就得到错误的结果。上述示例中,p1指向的d1显然小于p2指向的d2对象,但是Less内部并没有比较p1和p2指向的对象内容,而比较的是p1和p2指针的地址,这就无法达到预期而错误。

此时,就需要对模板进行特化。即:在原模板类的基础上,针对特殊类型所进行特殊化的实现方式。模板特化中分为函数模板特化与类模板特化。

换句话说就是特化对某些情况进行特殊处理。

2.2 函数模板特化

函数模板的特化步骤:

  1. 必须要先有一个基础的函数模板
  2. 关键字template后面接一对空的尖括号<>
  3. 函数名后跟一对尖括号,尖括号中指定需要特化的类型
  4. 函数形参表: 必须要和模板函数的基础参数类型完全相同,如果不同编译器可能会报一些奇怪的错误。

例如我们这里想让日期类按照日期比较
那我们就可以这样写

template<class T>
bool Less(T left, T right)
{return left < right;
}
// 对Less函数模板进行特化
template<>
bool Less<Date*>(Date* left, Date* right)
{return *left < *right;
}

所以特化就是特殊处理,例如这里T是Data*时就走下面的模版,其他情况就正常走上面的模版。

bool Less<Date*>(Date* left, Date* right)
{return *left < *right;
}

但是我们更推荐写一个函数来特殊处理,因为编译器在模版和现有的函数中会优先选择现有的函数。

2.3底层const

为什么前面我们不推荐用特殊解决呢?

template<class T>
bool Less(const T& left, const T& right)
{return left < right;
}
// 对Less函数模板进行特化
template<>
bool Less<Date*>(const Date*& left,  const Date*& right)
{return *left < *right;
}

这里我们特化的参数列表明明和原来的模版一模一样会什么还会报错呢?


所以当T是指针类型时,特化版本const要放在指针的右边。

同时如果是const指针我们的特化模版也匹配不上。

我们还需要特化一个const Date*的版本。


所以我们不太推荐函数模板的特化,更推荐直接写函数处理。

2.4 类模板特化

  • 全特化
    全特化即是将模板参数列表中所有的参数都确定化。
    格式如下:
template<class T1, class T2>//原模版
class Data
{
public:Data() { cout << "Data<T1, T2>" << endl; }
private:T1 _d1;T2 _d2;
};
template<>//全特化
class Data<int, char>
{
public:Data() { cout << "Data<int, char>" << endl; }
private:int _d1;char _d2;
};

这里下面的就会走全特化。

  • 偏特化
    偏特化:任何针对模版参数进一步进行条件限制设计的特化版本。比如对于以下模板类:
template<class T1, class T2>
class Data
{
public:Data() {cout<<"Data<T1, T2>" <<endl;}private:T1 _d1;T2 _d2;
};

偏特化有以下两种表现方式:

  • 部分特化
    将模板参数类表中的一部分参数特化。
// 将第二个参数特化为int
template <class T1>
class Data<T1, int>
{
public:Data() {cout<<"Data<T1, int>" <<endl;}
private:T1 _d1;int _d2;
};


需要注意的是,全特化和半特化都可以匹配时。
编译器会优先选择最匹配的,也就是全特化。

  • 参数更进一步的限制
    偏特化并不仅仅是指特化部分参数,而是针对模板参数更进一步的条件限制所设计出来的一个特化版本。
//两个参数偏特化为指针类型
template <typename T1, typename T2>
class Data <T1*, T2*>
{
public:Data() { cout << "Data<T1*, T2*>" << endl; }
private:T1 _d1;T2 _d2;
};
//两个参数偏特化为引用类型
template <typename T1, typename T2>
class Data <T1&, T2&>
{
public:Data() { cout << "Data<T1&, T2&>" << endl; }
private:T1 _d1;T2 _d2;
};

这里的两个偏特化就是针对指针类型和引用类型的特化。
也就是说只要是指针或引用就会匹配这两个特化版本。

同时指针和引用混在一起特化也可以。

所以之前我们deque日期类的指针比较仿函数就可以特化指针类型的。
这样就不需要传仿函数进去了。只要是特化就会自动走指针类型的比较
在这里插入图片描述
需要注意的是指针或引用的特化是按照特化版本的参数匹配而非原模版。
这样是为了更方便我们在类模版里面使用类型。

三.模板分离编译

3.1 什么是分离编译

一个程序(项目)由若干个源文件共同实现,而每个源文件单独编译生成目标文件,最后将所有目标文件链接起来形成单一的可执行文件的过程称为分离编译模式。

3.2 模板的分离编译

假如有以下场景,模板的声明与定义分离开,在头文件中进行声明,源文件中完成定义:

// a.h
template<class T>
T Add(const T& left, const T& right);
// a.cpp
template<class T>
T Add(const T& left, const T& right)
{return left + right;
}
// main.cpp
#include"a.h"
int main()
{Add(1, 2);Add(1.0, 2.0);return 0;
}

模版声明和定义会报链接错误。

这里模版声明和定义分离报错主要是没有实例化就不会生成汇编,就不会进入符号表。

  • 显示实例化
    所以我们向声明和定义分离时就显示实例化即可。
//显示实例化int类型
template//不需要<>
int Add(const int& left, const int& right);

但是我们更推荐将声明和定义都放在.h这种。

四.模版总结

  • 优点
    1.模板复用了代码,节省资源,更快的迭代开发,C++的标准模板库(STL)因此而产生
    2.增强了代码的灵活性
  • 缺点
    1.模板会导致代码膨胀问题,也会导致编译时间变长
    2.出现模板编译错误时,错误信息非常凌乱,不易定位错误

后言

这就是模版的特化及其编译分离。大家自己好好消化。今天就分享到这!感谢各位的耐心垂阅!咱们下期见!拜拜~

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

解决:无法在此设备上激活Windows因为无法连接到你的组织的激活服务器

问题&#xff1a; 桌面右下角会出现这个东西&#x1f447; 在设置里查看激活状态就会看到&#x1f447; 解决方法 &#xff1a; 1.打开CMD 搜索CMD&#xff0c;然后以管理员身份运行 2.设置 KMS服务器 1&#xff09;命令行输入&#xff1a; slmgr /skms kms.03k.org 然后…

1.6K+ Star!GenAIScript:一个可自动化的GenAI脚本环境

GenAIScript 简介 GenAIScript[1] 是一个 JavaScript-ish 环境&#xff0c;提供了便捷的工具用于文件摄入、提示开发和结构化数据提取。它允许用户以编程方式组装大型语言模型&#xff08;LLM&#xff09;的提示&#xff0c;并通过单一脚本协调 LLM、工具和数据。 项目特点 主…

高效管理iPhone存储:苹果手机怎么删除相似照片

在使用iPhone的过程中&#xff0c;我们经常会遇到存储空间不足的问题&#xff0c;尤其是当相册中充满了大量相似照片时。这些照片不仅占用了宝贵的存储空间&#xff0c;还可能使iPhone出现运行卡顿的情况。因此&#xff0c;我们迫切需要寻找苹果手机怎么删除相似照片的方法&…

TARE-PLANNER学习记录

参考&#xff1a; CMU-TARE 探索算法官方社区问答汇总_cmu localplanner 部署-CSDN博客 Tare_planner学习笔记_tare planner-CSDN博客 Tare_planner 学习教程(二)_tareplanner-CSDN博客 &#xff08;学习笔记&#xff09;机器人自主导航从零开始第七步——TARE Planner自主…

Moonshine - 新型开源ASR(语音识别)模型,体积小,速度快,比OpenAI Whisper快五倍 本地一键整合包下载

Moonshine 是由 Useful Sensors 公司推出的一系列「语音到文本&#xff08;speech-to-text, STT&#xff09;转换模型」&#xff0c;旨在为资源受限设备提供快速而准确的「自动语音识别&#xff08;ASR&#xff09;服务」。Moonshine 的设计特别适合于需要即时响应的应用场景&a…

【实验八】前馈神经网络(4)优化问题

1 参数初始化 模型构建 模型训练 优化 完整代码 2 梯度消失问题 模型构建 模型训练 完整代码 3 死亡Relu问题 模型构建 模型训练 优化 完整代码 1 参数初始化 实现一个神经网络前&#xff0c;需要先初始化模型参数。如果对每一层的权重和偏置都用0初始化&#xff0…

华为-宝塔-MongoDB无法登录

1、宝塔防火墙服务器安全组放开端口号 2、用数据库对应的用户名和密码登录 2-1&#xff1a;不指定验证数据库时用root账号密码登录 2-2&#xff1a;如果设置了验证数据库就用验证数据库对应的账号和密码登录

Scala入门基础(16)scala的包

Scala的包定义包定义包对象Scala的包的导入导入重命名 一.Scala的包 package&#xff08;包&#xff1a;一个容器。可以把类&#xff0c;对象&#xff0c;包&#xff0c;装入。 好处&#xff1a; 区分同名的类&#xff1b;类很多时&#xff0c;更好地管理类&#xff1b;控制…

Android IPC机制(一)多进程模式

1. 什么是进程&#xff1f; 进程是操作系统分配资源&#xff08;如 CPU、内存等&#xff09;的基本单位。简单来说&#xff0c;进程是一个正在执行的程序的实例。每个进程都有自己的内存空间、数据栈和其他辅助数据&#xff0c;用于跟踪进程的执行状态。在 Android 中&#xff…

【笔记】铜导线在高频下的损耗

参考资料&#xff1a;Litz Wire: Practical Design Considerations for Todays High Frequency Applications&#xff0c;kyle jensen,2020 1.高频条件下因为集肤效应&#xff0c;需要选择多股线 否则高频下因为集肤效应和接近效应&#xff0c;所引发的交流阻抗上升&#xff…

火语言RPA流程组件介绍--指纹浏览器管理

&#x1f6a9;【组件功能】&#xff1a;指纹浏览器配置管理创建、删除、判断是否存在 配置预览 配置说明 操作类型 有“创建、删除、判断是否存在”3种类型供选择。 指纹浏览器配置名称 支持T或# 默认FLOW输入项 填写指纹环境分身名称。 操作方式 有“名称、Id”2种方式…

windows自启动 映像劫持 屏保

Windows权限维持—自启动&映像劫持&粘滞键&辅助屏保后门 自启动 自启动路径加载 受控windows机器选择当前用户C盘目录下将文件放到这里每到电脑服务器重启就会自动加这次路径下文件 C:\Users\月\AppData\Roaming\Microsoft\Windows\Start Menu\Programs\Startu…

SSH实验3拒绝root用户远程登录

打开配置文件&#xff1a; 默认为root用户密码登录&#xff1a; 加一行PermitRootLogin no&#xff0c;拒绝登录&#xff1a; 再打开这个配置文件&#xff1a; yes改为no&#xff1a; 查看SELinux 当前处于宽松模式&#xff0c;并且关闭防火墙&#xff1a; 重启sshd&#xff1a…

mysql 主从复制

一、通过二进制文件binlog进行主从同步 开启防火墙3306端口 1.设置主服务器&#xff1a;/etc/my.cnf log-binbinlog #二进制文件名称&#xff0c;需要开启 binlog-formatROW; # row,statement,mixed mysql默认采用statement statement&#xff1a;主要记录了sql。日志…

Python作业记录

复制过来的代码的换行有问题&#xff0c;但是也不是什么大问题。 后续我会进行补充和修改。 请将如下英文短句根据单词切分成列表&#xff1a; The continent of Antarctica is rising. It is due to a geological phenomenon called post-glacial uplift 并在切分好的列表…

Spring Boot框架下的水电管理系统开发

1系统概述 1.1 研究背景 随着计算机技术的发展以及计算机网络的逐渐普及&#xff0c;互联网成为人们查找信息的重要场所&#xff0c;二十一世纪是信息的时代&#xff0c;所以信息的管理显得特别重要。因此&#xff0c;使用计算机来管理大学城水电管理系统的相关信息成为必然。开…

vue3+less使用主题定制(多主题定制)可切换主题

假如要使用两套主题&#xff1a;蓝色、红色 例如&#xff1a; 首先确保自己的vue3项目有less&#xff0c;这边不多做接入解释 1、在src目录下建一个styles文件夹&#xff0c;在syles文件夹下面新建两个less文件&#xff1a;theme.less和variables.less&#xff1b; theme.le…

PyQt5实战——翻译器的UI页面设计以及代码实现(七)

个人博客&#xff1a;苏三有春的博客 系类往期文章&#xff1a; PyQt5实战——多脚本集合包&#xff0c;前言与环境配置&#xff08;一&#xff09; PyQt5实战——多脚本集合包&#xff0c;UI以及工程布局&#xff08;二&#xff09; PyQt5实战——多脚本集合包&#xff0c;程序…

【种完麦子,我就往南走,去西双版纳,过个冬天!】

麦子奶奶&#xff1a;冰哥&#xff0c;你好。 大冰&#xff1a;你好&#xff0c;咱俩不定谁大呢。 麦子奶奶&#xff1a;嗯&#xff0c;我大&#xff0c;我60多了&#xff0c;你各方面都是哥。 大冰&#xff1a;阿姨好 麦子奶奶&#xff1a;我想出去看看祖国的大好河山&…

koa + sequelize做距离计算(MySql篇)

1.核心思路 1.利用sequelize的fn方法调用MySql原生函数&#xff08;st_distance_sphere、point&#xff09; 2.这里利用到了MySql的原生函数&#xff0c;不懂可以去看看mysql的函数知识 2.核心代码 //st_distance_sphere、point函数用来计算当前经纬度和目的地经纬度 //col…