c++ | 模板进阶

  前言

本篇博客讲解c++中的模板的一些其他知识

💓 个人主页:普通young man-CSDN博客

⏩ 文章专栏:C++_普通young man的博客-CSDN博客

⏩ 本人giee:   普通小青年 (pu-tong-young-man) - Gitee.com

      若有问题 评论区见📝

🎉欢迎大家点赞👍收藏⭐文章

目录

概述

1. 非类型模板参数

2. 模板的特化

2.1 函数模板特化

2.2 类模板特化

2.2.1 全特化

2.2.2 偏特化

 类模板特化应用示例

3. 模板分离编译

故事背景

分离编译模式的应用

解决方法

总结


概述

C++模板是一种强大的功能,它允许开发者编写泛型代码,从而提高代码的重用性和灵活性。本文将探讨非类型模板参数的概念及其限制,以及模板特化的方法,包括函数模板特化和类模板特化。

1. 非类型模板参数

非类型模板参数是用一个常量作为类(函数)模板的一个参数,在类(函数)模板中可将该参数当成常量来使用。非类型模板参数必须在编译期就能确认结果,并且有一些限制,例如:

  • 浮点数、类对象以及字符串不允许作为非类型模板参数。
  • 非类型的模板参数必须在编译期就能确认结果。
template<class T,size_t size = 10>
int add(T a1) {return a1 + size;}template<class T,size_t N = 10>
class MyClass
{
public:MyClass(const T& val):arr[0] = val;{}
private:int arr[N];
};void test1() {//cout << add<>(10) << endl;MyClass<int> s1(10);}
int main() {test1();return 0;
}

2. 模板的特化

模板特化是在原模板类的基础上,针对特殊类型所进行特殊化的实现方式。模板特化分为函数模板特化与类模板特化。

2.1 函数模板特化

函数模板特化是当函数模板在处理某些特定类型时可能无法得到预期结果的情况下使用的。特化过程包括以下几个步骤:

  1. 必须要先有一个基础的函数模板。
  2. 使用关键字 template 后面接一对空的尖括号 <>
  3. 在函数名后跟一对尖括号,尖括号中指定需要特化的类型。
  4. 函数形参表必须与模板函数的基础参数类型完全相同。
class Date
{
public:Date(int year = 1900, int month = 1, int day = 1): _year(year), _month(month), _day(day){}bool operator<(const Date& d)const{return (_year < d._year) ||(_year == d._year && _month < d._month) ||(_year == d._year && _month == d._month && _day < d._day);}bool operator>(const Date& d)const{return (_year > d._year) ||(_year == d._year && _month > d._month) ||(_year == d._year && _month == d._month && _day > d._day);}friend ostream& operator<<(ostream& _cout, const Date& d){_cout << d._year << "-" << d._month << "-" << d._day;return _cout;}private:int _year;int _month;int _day;
};class DateLess
{
public://bool operator()(Date* p1, Date* p2)//{//	return *p1 < *p2;//}
};//模板特化
template<class T>
bool lessfunc(const T& left, const  T& right) {return left < right;
}//特化--不推荐
//template<>
//bool lessfunc<Date*>(Date* const& left, Date* const& right)
//{
//	return *left < *right;
//}
//
//template<>
//bool lessfunc<const Date*>(const Date* const& left,const Date* const& right)
//{
//	return *left < *right;
//}//推荐直接写成函数
bool lessfunc(Date* p1, Date* p2) {return *p1 < *p2;
}
bool lessfunc(const Date* p1,const Date* p2) {return *p1 < *p2;
}void test2() {//cout << lessfunc(1, 2) << endl;//int a = 100, b = 20;//int* pa = &a;//int* pb = &b;//cout << lessfunc(pa,pb) << endl;//比较错误Date d1(2025, 7, 20);Date d2(2022, 7, 8);cout << lessfunc(d1, d2) << endl;Date* p1 = &d1;Date* p2 = &d2;cout << lessfunc(p1, p2) << endl;//比较错误}
int main() {test2();return 0;
}

这里为什么不推荐写成函数特化,就是因为易读性不好,写函数的话直接一点,简单易操作
 

  1. 代码冗余:特化版本的函数可能需要重复原始模板函数的部分逻辑,这可能导致代码冗余。
  2. 理解难度:特化版本的存在可能会使代码的逻辑变得模糊不清,尤其是当多个特化版本同时存在时,读者需要花费更多的时间去理解哪个版本会被调用。
  3. 调试困难:当出现问题时,调试特化版本可能比调试普通函数更加困难,因为需要考虑特化版本与模板之间的交互。
  4. 过度特化:过度使用特化可能导致代码膨胀,并且增加维护成本。

2.2 类模板特化

类模板特化分为全特化和偏特化。

2.2.1 全特化

全特化是将模板参数列表中所有的参数都确定化。

//全特化
template<>
class Data<int,int>
{
public:Data() { cout << "data<int,int>" << endl; }};
2.2.2 偏特化

偏特化是指对模板参数进行进一步的条件限制设计出的特化版本。偏特化有两种表现形式:部分特化和参数更进一步的限制。

//偏特化/半特化//全特化
template<class T1>
class Data<T1, int>
{
public:Data() { cout << "data<T1,int>" << endl; }};

参数更进一步的限制 是针对模板参数的类型进行更严格的限制。例如,将两个参数特化为指针类型:

template<>
class Data<int*, int*>
{
public:Data() { cout << "data<int*,int*>" << endl; }};template<>
class Data<int&, int&>
{
public:Data() { cout << "data<int&,int&>" << endl; }};

 类模板特化应用示例

我们可以通过一个例子来看一下如何使用类模板特化来解决特定的问题。假设我们有一个用于比较的类模板 Less,它可以用于直接比较日期对象,但是当比较指针时则会比较指针的地址而非指针指向的内容。我们可以使用类模板特化来解决这个问题。

#include<algorithm>
#include<vector>
class Date
{
public:Date(int year = 1900, int month = 1, int day = 1): _year(year), _month(month), _day(day){}bool operator<(const Date& d)const{return (_year < d._year) ||(_year == d._year && _month < d._month) ||(_year == d._year && _month == d._month && _day < d._day);}bool operator>(const Date& d)const{return (_year > d._year) ||(_year == d._year && _month > d._month) ||(_year == d._year && _month == d._month && _day > d._day);}friend ostream& operator<<(ostream& _cout, const Date& d){_cout << d._year << "-" << d._month << "-" << d._day;return _cout;}private:int _year;int _month;int _day;
};template<class T>
class Less {
public://仿函数bool operator()(const T& a1,const T& a2) {return a1 > a2;}
};//特化
template<class Data>
class Less <Data*>{
public://仿函数bool operator()(Data* a1, Data* a2) const{return *a1 > *a2;}
};int main() {Date d1(2024, 1, 1);Date d2(2024, 11, 1);Date d3(2024, 10, 1);//vector<Date> s1;//s1.push_back(d1);//s1.push_back(d2);//s1.push_back(d3);//sort(s1.begin(), s1.end(), Less<Date>());vector<Date*> s1;s1.push_back(&d1);s1.push_back(&d2);s1.push_back(&d3);sort(s1.begin(), s1.end(), Less<Date*>());return 0;
}

其实就可以发现,特化一般是在一些特殊情况进行一个特殊处理

3. 模板分离编译

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

在使用模板时,如果模板的声明与定义分离开,在头文件中进行声明,源文件中完成定义,则需要特别注意编译过程。解决方法包括:

  1. 将声明和定义放到一个文件 "xxx.hpp" 或者 "xxx.h" 中。这是推荐的做法。
  2. 显式实例化模板定义的位置。这种方法不实用,不推荐使用。

故事背景

假设你正在筹备一场盛大的聚会,你需要向不同的供应商订购所需物品和服务。为了组织好这场聚会,你决定采用一种高效的协调方式——分离编译模式。

  1. 制定采购清单:首先,你需要制定一份采购清单,列出所有需要的物品和服务。这份清单相当于模板的声明,它定义了你需要什么,但没有具体到哪家供应商。

  2. 发送询价单:接下来,你将向各个供应商发送询价单,这些询价单包含了具体的数量和要求。每份询价单相当于模板的具体定义,它们包含了特定的实现细节。

  3. 供应商报价:供应商会根据你的询价单进行报价。这个过程类似于编译阶段,供应商评估需求并给出价格。

  4. 签订合同:如果供应商的报价符合你的预算,你们将签订合同。合同相当于生成的目标文件,它明确了双方的责任和义务。

  5. 完成交易:最后,供应商会根据合同提供货物和服务,你则支付费用。这相当于链接阶段,将所有合同(目标文件)组合在一起形成一个完整的交易过程。

分离编译模式的应用

现在,让我们看看如何将这个故事映射到模板分离编译上:

  • 声明和定义放在同一个文件:这就像你将采购清单和询价单合并到一起,直接交给供应商。这种方式简单明了,因为供应商不需要在不同的地方查找你的清单和询价单详情。同样地,在编程中,如果你将模板的声明和定义放在同一个文件(例如 xxx.hpp)中,那么编译器可以在一个地方找到所有需要的信息,无需额外的链接步骤。

  • 显式实例化:这类似于你在采购清单中明确指出你需要从哪家供应商订购多少数量的商品,并且直接发送询价单。在编程中,这意味着你需要在源文件中显式地实例化模板,告诉编译器你需要哪些具体类型的实例。然而,这种方法在实际操作中并不常见,因为它增加了额外的工作量并且可能导致代码冗余。

这边举了一个例子来更好的理解为什么声明和定义放在两个文件会报错

解决方法

1.模板的声明和定义都在同一个文件(推荐)

2.显示实例化<这一步就是在链接的时候告诉编译器类型>(不推荐,容易代码膨胀)

这边的template不加<>是为了和特化区分(可以理解为文件在链接的时候告诉它这个文件里的模板类型)

总结

模板是C++语言的重要特性,它提供了代码重用的强大手段。然而,使用模板也需要注意其局限性,如非类型模板参数的限制、模板特化的正确使用等。通过合理利用模板特化,我们可以解决特定类型的问题,使得代码更加灵活和高效。

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

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

相关文章

推荐一个国内Midjourney镜像站,限时充值享5折优惠 结尾附实测图片

作为一名绘画爱好者&#xff0c;你是否曾梦想过将脑海中的画面转化为现实&#xff1f;现在&#xff0c;有了群嘉智创平台&#xff08;ai.qunzjia.cn&#xff09;&#xff0c;这一切都将成为可能。群嘉智创是国内领先的AI对话与Midjourney绘画服务平台&#xff0c;通过接入国内多…

[Meachines] [Easy] Legacy nmap 漏洞扫描脚本深度发现+MS08-067

信息收集 IP AddressOpening Ports10.10.10.4TCP:135,139,445 $ nmap -p- 10.10.10.4 --min-rate 1000 -sC -sV -Pn PORT STATE SERVICE VERSION 135/tcp open msrpc Microsoft Windows RPC 139/tcp open netbios-ssn Microsoft Windows n…

论文阅读1 Scaling Synthetic Data Creation with 1,000,000,000 Personas

Scaling Synthetic Data Creation with 1,000,000,000 Personas 链接&#xff1a;https://github.com/tencent-ailab/persona-hub/ 文章目录 Scaling Synthetic Data Creation with 1,000,000,000 Personas1. 摘要2. 背景2.1 什么是数据合成2.2 为什么需要数据合成2.3 10亿种人…

基于Kotlin Multiplatform的鸿蒙跨平台开发实践

一、 背景 在 2023 年的华为开发者大会&#xff08;HDC&#xff09;上&#xff0c;华为预告了一个全新的鸿蒙系统 Harmony Next 版本。与之前的鸿蒙系统不同&#xff0c;Harmony Next完全摒弃了对 AOSP 的兼容&#xff0c;彻底基于 OpenHarmony 开源鸿蒙实现。这意味着该系统将…

在idea中的git选择某一次记录拉出一个新分支

一 创建新分支 1.1 操作步骤 需求&#xff1a;需要在图中标红的历史记录&#xff0c;从此记录拉出一个分支 1.右键【new branch】 2.起一个新的名字&#xff1a; 3.新分支代码

《图解设计模式》笔记(四)分开考虑

九、Bridge模式&#xff1a;将类的功能层次结构与实现层次结构分离 类的两个层次结构和作用 类的功能层次结构&#xff1a;希望增加新功能时 父类有基本功能&#xff0c;在子类中增加新功能 Something父类 …├─SomethingGood子类 想要再增加新功能 Something父类 …├─So…

LeetCode.55.跳跃游戏(贪心算法思路)

题目描述&#xff1a; 给你一个非负整数数组 nums &#xff0c;你最初位于数组的 第一个下标 。数组中的每个元素代表你在该位置可以跳跃的最大长度。 判断你是否能够到达最后一个下标&#xff0c;如果可以&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 输…

Docker容器镜像及其打包

容器镜像分类 1. 系统类镜像 2. 应⽤镜像 搜索镜像 # 默认docker.hub docker search centos 下载镜像 docker pull centos 默认下载最新版本 1. 打包 [rootdocker001 ~]# systemctl start docker.service [rootdocker001 ~]# docker save -o centos.tar centos:latest [root…

基于SpringBoot的线上教学平台系统

你好呀&#xff0c;我是计算机学姐码农小野&#xff01;如果有相关需求&#xff0c;可以私信联系我。 开发语言 Java 数据库 MySQL 技术 SpringBoot框架&#xff0c;Java语言 工具 IDEA/Eclipse、Navicat、Maven 系统展示 首页 管理员功能模块 学员功能模块 前台首页…

Ozon在奥伦堡州开设首个配送中心,Ozon还有机会赚钱吗?

Ozon平台成立于1998年&#xff0c;是俄罗斯唯一上市的B2C电商平台&#xff0c;在俄罗斯电商市场中占据着到达62%的市场份额&#xff0c;具有强大的市场影响力和吸引力。Ozon拥有数千万的活跃用户&#xff0c;覆盖了俄罗斯各个年龄段和消费层次的群体&#xff0c;而且Ozon拥有俄…

“精准学”官宣将公布中国首个语音端到端大模型

教育科技公司“精准学”宣布&#xff0c;公司已在AI语音交互技术上取得领先性的突破&#xff0c;成功训练了中国首个语音端到端大模型“心流知镜-s(V02)”&#xff0c;可直接实现语音输入-语音输出的交互&#xff0c;使其更适配辅学场景&#xff0c;使大模型达到“真人老师”级…

当《黑神话:悟空》遇上openKylin,国产力量的极致碰撞!

了解更多银河麒麟操作系统全新产品&#xff0c;请点击访问 麒麟软件产品专区&#xff1a;https://product.kylinos.cn 开发者专区&#xff1a;https://developer.kylinos.cn 文档中心&#xff1a;https://documentkylinos.cn 万众瞩目的国产3A游戏巨作《黑神话&#xff1a;悟…

【人工智能】如何在白嫖的阿里云PAI平台上跑模型?

在“交互式建模&#xff08;DSW&#xff09;”中新建实例&#xff0c;阿里云自带的示例镜像是很少的&#xff0c;所以我们只需要筛选出适合你的项目的CUDA版本就好。DSW实例可以看作是一个Linux虚拟机&#xff0c;之后我们在实例中新建另一个Python环境使用即可。 新建完实例后…

DevExpress中Blazor部分学习

DevExpress中Blazor学习 1 DevExpress版本2 学习步骤2.1 查看Dev相应的Demo2.2 创建第一个相关应用2.3 使用XPO进行相关数据操作2.4 Dev Blazor使用XPO操作 3 学习中遇到问题及解决方案3.1 打开Dev相关Demo报错 1 DevExpress版本 安装较新的DevExpress&#xff0c;我这边使用的…

基于FreeRTOS的STM32多功能手表

前言 项目背景 项目演示 使用到的硬件 项目原理图 目前版本实现的功能 设计到的freertos知识 实现思路 代码讲解 初始化GPIO引脚、配置时钟 蜂鸣器初始化以及软件定时器创建 系统默认创建的defaultTaskHandle 创建七个Task&#xff0c;代表七个功能 ShowTimeTask …

京东软件测试岗面试题(干货)含答案+文档

&#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 前面看到了一些面试题&#xff0c;总感觉会用得到&#xff0c;但是看一遍又记不住&#xff0c;所以我把面试题都整合在一起&#xff0c;都是来自各路大佬的分享&am…

自然语言处理系列三十三》 语义相似度》同义词词林》算法原理

注&#xff1a;此文章内容均节选自充电了么创始人&#xff0c;CEO兼CTO陈敬雷老师的新书《自然语言处理原理与实战》&#xff08;人工智能科学与技术丛书&#xff09;【陈敬雷编著】【清华大学出版社】 文章目录 自然语言处理系列三十三同义词词林算法原理代码实战 总结 自然语…

LLama 3 跨各种 GPU 类型的基准测试

2024 年 4 月 18 日&#xff0c;AI 社区对 Llama 3 70B 的发布表示欢迎&#xff0c;这是一款最先进的大型语言模型 &#xff08;LLM&#xff09;。该型号是 Llama 系列的下一代产品&#xff0c;支持广泛的用例。该模型 istelf 在广泛的行业平台上表现良好&#xff0c;并提供了新…

基于STM32开发的智能室内照明系统

目录 引言环境准备工作 硬件准备软件安装与配置系统设计 系统架构硬件连接代码实现 系统初始化光照强度监测与处理照明控制与状态指示Wi-Fi通信与远程控制应用场景 智能家居照明管理办公室和公共场所的智能照明常见问题及解决方案 常见问题解决方案结论 1. 引言 随着智能家居…

探索地理空间分析的新世界:Geopandas的魔力

文章目录 探索地理空间分析的新世界&#xff1a;Geopandas的魔力背景&#xff1a;为何选择Geopandas&#xff1f;这个库是什么&#xff1f;如何安装这个库&#xff1f;五个简单的库函数使用方法场景应用&#xff1a;Geopandas在实际工作中的应用常见bug及解决方案总结 探索地理…