初识C++ · 模板进阶

news2024/11/18 1:33:12

目录

前言:

1 非类型模板参数

2 按需实例化

3 模板特化

4 模板的分离编译


前言:

前面模板我们会了简单的使用,这里带来模板的进阶,当然,也就那么几个知识点,并不太难。


1 非类型模板参数

先来看这样一段代码:

#define N 100
template<class T>
class Arr
{
public:

private:
	T _arr[N];
};

如果我们想要创建一个整型数组,可以使用这个类来创建,但是我们面临一个问题就是该数组的大小是固定的,我们想要简单控制这个数组的大小,可以使用宏,但是还是不够简便,因为宏不方便调试不说,实际上也是指定了大小,那么我们想要使用一个类,来创建不同大小的数组该怎么办?
这里使用到的就是非类型模板参数,如下:

template<class T,size_t N = 100>
class Arr
{
public:

private:
	T _arr[N];
};
int main()
{
	Arr<int,10> a1;
	Arr<int,1000> a2;
	Arr<int> a3;
	return 0;
}

这里就得到了我们想要的不同大小的数组,那么,来个问题,编译器一共实现了几个类?

答:编译器这里一共实现了3个类,编译器根据模板参数的不同,就实现了不同的类。这里的非类型模板参数,我们可以理解为常量,如这里的N,但是在C++11只支持整型,连浮点数都不可以,只支持整型,比如int size_t char一类的,在C++ 20之后才可以支持其他类型。

这里涉及到了数组,那么引入一个小的知识点就是对于越界来说,array 数组 vector有着不同的反应:

int main()
{
	int arr[10];
	arr[10];
	arr[15] = 1;
	return 0;
}

对于普通数组来说,普通的越界只读来说,比如arr[10]是检查不来错误的,越界写来说,也是很多抽查不出来的,比如这段代码在vs2019上就不会报错。

那么对于array来说:

int main()
{
	std::array<int,10> array;
	array[10];
	return 0;
}

任何读写越界都会报错,但是呢,这是c++委员会后面加的,但是挺鸡肋的,因为我们有vector。

int main()
{
	std::vector<int> v;
	v.reserve(1000);
	return 0;
}

vector对越界的读写都会报错,这是一方面,其次是array是静态的数组,也就是大小定了,并且,它所属的空间是栈,栈的空间相对堆来说就会小很多,所以面临开大空间的时候,array就不吃香了,vector没事,因为可以动态开辟。


2 按需实例化

先看这样一段代码:

template<class T,size_t N = 100>
class Arr
{
public:
	T& operator[](size_t i)
	{
		size(1);
		return _arr[i];
	}
	size_t size()
	{
		return _size;
	}
	bool empty()
	{
		return _size == 0;
	}
private:
	T _arr[N];
	size_t _size = 0;
};
int main()
{
	Arr<int> a1;
	a1.empty();

	return 0;
}

从语法层面来说,size()函数没有参数,那么我们传参数的话就会导致报错对吧?可是,实际上:

代码是没有报错,也就是说size()传参数是对的吗?

不,这是因为按需实例化

在主函数里面,我们实例化了a1,并且调用了empty函数,但是我们没有调用operator[]函数,那么编译器就不会实例化operator函数,因为我们没有调用,既然没有实例化size函数,那么传什么都不会报错,这就是按需实例化

再细节一点来说,编译器会根据模板实例化->实例化一个半成品模板->再实例化为一个具体的类或者函数->最后才是语法编译,所以没有语法报错。


3 模板特化

特化我们可以理解为特殊化处理,比如我们在栈和队列的时候实现的日期类的比较,就可以不用仿函数来实现比较,可以用特化来处理。

日期类:

class Date
{
public:
	friend ostream& operator<<(ostream& _cout, const Date& d);

	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);
	}
private:
	size_t _year;	
	size_t _month;
	size_t _day;
};
ostream& operator<<(ostream& _cout, const Date& d)
{
	_cout << d._year << "-" << d._month << "-" << d._day;
	return _cout;
}
template<class T>
bool less(const T& a,const T& b)
{
	return a < b;
}
int main()
{
	Date d1(2024, 5, 20);
	Date d2(2024, 5, 21);
	cout << (d1 < d2) << endl;
	return 0;
}

比如,使用函数模板,对于重载了比较符号的比较是没问题的,如果使用指针就会报错,因为默认是按照指针比较的,这里就可以使用特化:

int main()
{
	Date* p1 = new Date(2020, 1, 1);
	Date* p2 = new Date(2020, 1, 2);
	cout << (p1 < p2) << endl;
	return 0;
}

这段代码是有问题的是不用多说的,下面是解决方案:

template<class T>
bool Less(T left, T right)
{
	cout << "bool Less(T left, T right)" << endl;
	return left < right;
}

template<>
bool Less<Date*>(Date* p1,Date* p2)
{
	return *p1 < *p2;
}

int main()
{
	Date d1(2022, 7, 7);
	Date d2(2022, 7, 8);
	Date* p1 = &d1;
	Date* p2 = &d2;
	cout << Less(p1, p2) << endl;
	return 0;
}

这里的语法就比较怪了,现在使用的是函数模板,有点像函数重载的感觉,当然,我们也可以直接重载一个出来:

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

那么,调用是怎么调用的呢?

这里就和半成品,成品是一个道理,重载的函数就相当于成品,特化的函数快成品了,模板就是个半成品,调用的顺序也就说的通了。

以上是函数模板的特化,看起来就像是函数重载,接着是类中的模板特化:


//普通
template<class T1,class T2>
class Data
{
public:
	Data() { cout << "Data<T1, T2>" << endl; }

};

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

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

使用方式和函数模板其实是差不多的,在特化这里分为全特化和偏特化,特化也不是什么特别的东西,其实就是对参数的进一步限制而已。


4 模板的分离编译

使用模板的时候,定义和声明最好放在一个文件,.h和.c文件分离会报错的,这里简单举个例子:

// 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;
}

在调用这个函数的时候就会报错,只需要想清楚一个简单的问题就可以了,两个T是不是一样的T,能否用.h文件里面的T去平替.cpp里面的T,当然是不可以的,所以这里,就会报错,报的是链接错误,.h文件编译成功后,.cpp里面的文件是没有编译好的,因为T不知道是什么类型,调用的时候就会报错。

在链接阶段,编译器按照修饰好之后的函数名在符号表里面寻找函数,不知道类型就没有生成修饰好的函数名,那么就会报如下类似的错:


感谢阅读!

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

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

相关文章

C语言过度C++语法补充(面向对象之前语法)

目录 1. C相较于C语言新增的语法 0. C 中的输入输出 1. 命名空间 1. 我们如何定义一个命名空间&#xff1f; 2. 如何使用一个命名空间 3. 命名空间中可以定义什么&#xff1f; 4. 在 相同或者不同 的文件中如果出现 同名的命名空间 会如何&#xff1f; 5. 总结~~撒花~~…

大模型基础——从零实现一个Transformer(1)

一、Transformer模型架构图 主要模块&#xff1a; embedding层&#xff1a; Input/Output Embedding&#xff1a; 将每个标记(token)转换为对应的向量表示。 Positional Encoding&#xff1a;由于没有时序信息&#xff0c;需要额外加入位置编码。 N个 block堆叠: Multi-Head …

vue不同页面切换的方式(Vue动态组件)

v-if实现 <!--Calender.vue--> <template><a-calendar v-model:value"value" panelChange"onPanelChange" /></template> <script setup> import { ref } from vue; const value ref(); const onPanelChange (value, mod…

Oracle EBS AP发票验证-计税期间出现意外错误解决方法

系统版本 RDBMS : 12.1.0.2.0 Oracle Applications : 12.2.6 问题症状: **打开发票题头或发票行“税详细信息”**错误提示如下: 由于以下原因而无法针对"税"窗口中所做的修改更新 Oraclee Payables信息: 尚未为税率或帐户来源税率设置可退回税/应纳税额帐户。请…

【Linux】The server quit without updating PID file的几种解决方案

&#x1f60e; 作者介绍&#xff1a;我是程序员洲洲&#xff0c;一个热爱写作的非著名程序员。CSDN全栈优质领域创作者、华为云博客社区云享专家、阿里云博客社区专家博主。 &#x1f913; 同时欢迎大家关注其他专栏&#xff0c;我将分享Web前后端开发、人工智能、机器学习、深…

简单聊下服务器防病毒

在当今数字化时代&#xff0c;服务器作为数据存储、处理与传输的核心设备&#xff0c;其安全性显得尤为关键。服务器防病毒工作&#xff0c;不仅是保障企业信息安全的重要一环&#xff0c;更是维护用户数据隐私的关键举措。以下&#xff0c;我们将从多个方面&#xff0c;简单探…

29网课交单平台 epay.php SQL注入漏洞复现

0x01 产品简介 29网课交单平台是一个专注于在线教育和知识付费领域的交单平台。该平台基于PHP开发,通过全开源修复和优化,为用户提供了高效、稳定、安全的在线学习和交易环境。作为知识付费系统的重要组成部分,充分利用了互联网的优势,为用户提供了便捷的支付方式、高效的…

Java使用OpenCV计算两张图片相似度

业务&#xff1a;找出两个表的重复的图片。 图片在表里存的是二进制值&#xff0c;存在大量由于一些特殊情况例如扫描有差异&#xff0c;导致图片存的二进制值不同&#xff0c;但图片其实是一样来的。 所以找出两个表重复相同的图片&#xff0c;不可能只是单纯的比较二进制值…

若依RuoYi-Vue分离版—增加通知公告预览及缩放功能

若依RuoYi-Vue分离版—增加通知公告预览及缩放功能 前言开发通知公告 前言 若依分离版的通知公告没有预览功能&#xff0c;想开发通知公告功能 开发通知公告 效果如下 具体开发内容 修改若依notice代码如下。 <template><div class"app-container"&g…

三十五篇:数字化转型的引擎:赋能企业的ERP系统全景

数字化转型的引擎&#xff1a;赋能企业的ERP系统全景 1. 引言 在这个快速变化的数字时代&#xff0c;现代企业面临着前所未有的挑战和机遇。企业资源计划&#xff08;ERP&#xff09;系统&#xff0c;作为数字化转型的核心&#xff0c;扮演着至关重要的角色。它不仅是企业运营…

霸气的短视频:成都科成博通文化传媒公司

霸气的短视频&#xff1a;瞬间的力量与魅力 在数字化浪潮中&#xff0c;短视频以其独特的魅力迅速崛起&#xff0c;成为社交媒体的新宠。而在众多短视频中&#xff0c;那些充满霸气、让人热血沸腾的作品&#xff0c;总能引起广泛的关注和讨论。成都科成博通文化传媒公司将从内…

Linux文本处理三剑客之awk命令

官方文档&#xff1a;https://www.gnu.org/software/gawk/manual/gawk.html 什么是awk&#xff1f; Awk是一种文本处理工具&#xff0c;它的名字是由其三位创始人&#xff08;Aho、Weinberger和Kernighan&#xff09;的姓氏首字母组成的。Awk的设计初衷是用于处理结构化文本数…

SSM整合总结

一.核心问题 (一)两个容器 web容器 web相关组件&#xff08;controller,springmvc核心组件&#xff09; root容器 业务和持久层相关组件&#xff08;service,aop,tx,dataSource,mybatis,mapper等&#xff09; 父容器&#xff1a;root容器&#xff0c;盛放service、mapper、…

前端 JS 经典:打印对象的 bug

1. 问题 相信这个 console 打印语句的 bug&#xff0c;其实小伙伴们是遇到过的&#xff0c;就是你有一个对象&#xff0c;通过 console&#xff0c;打印一次&#xff0c;然后经过一些处理&#xff0c;再通过 console 打印&#xff0c;发现两次打印的结果是一样的&#xff0c;第…

算法金 | 这次终于能把张量(Tensor)搞清楚了!

大侠幸会&#xff0c;在下全网同名[算法金] 0 基础转 AI 上岸&#xff0c;多个算法赛 Top [日更万日&#xff0c;让更多人享受智能乐趣] 1. 张量&#xff08;Tensor&#xff09;基础概念 1.1 张量的定义与重要性 张量是深度学习中用于表示数据的核心结构&#xff0c;它可以视…

国标GB/T 28181详解:国标GBT28181-2022的客户端主动发起历史视音频回放流程

目录 一、定义 二、作用 1、提供有效的数据回顾机制 2、增强监控系统的功能性 3、保障数据传输与存储的可靠性 4、实现精细化的操作与控制 5、促进监控系统的集成与发展 三、历史视音频回放的基本要求 四、命令流程 1、流程图 2、流程描述 五、协议接口 1、会话控…

Paper速读-[Visual Prompt Multi-Modal Tracking]-Dlut.edu-CVPR2023

文章目录 简介关于具体的思路问题描述算法细节 实验结果模型的潜力模型结果 论文链接&#xff1a;Visual Prompt Multi-Modal Tracking 开源代码&#xff1a;Official implementation of ViPT 简介 这篇文章说了个什么事情呢&#xff0c;来咱们先看简单的介绍图 简单来说&am…

从零开始学习Linux(9)----文件系统

目录 1.前言 1.铺垫 2.C语言文件接口-对比重定向 3.文件fd 4.缓冲区问题 2.文件系统 1.磁盘的物理存储 2.磁盘的逻辑存储 3.软硬链接 1.硬链接 2.软链接 1.前言 1.铺垫 a.文件内容属性 b.访问文件之前&#xff0c;都得先打开&#xff0c;修改文件&#xff0c;都是通…

ElementUI中date-picker组件,怎么把大写月份改为阿拉伯数字月份(例如:一月、二月,改为1月、2月)

要将 Element UI 的 <el-date-picker> 组件中的月份名称从中文大写&#xff08;如 "一月", "二月"&#xff09;更改为阿拉伯数字&#xff08;如 "1月", "2月"&#xff09;&#xff0c;需要进行一些定制化处理。可以通过国际化&a…

LIUNX系统编程:信号(3)

目录 3.信号的处理 3.1信号是什么时候被处理的 read系统调用 3.2信号是怎样被处理的 内核态和用户态 3.3操作系统是如何运行处理信号的呢&#xff1f; 中断技术 什么让操作系统运行起来的 3.4捕捉信号的其他方式 ​编辑 demo代码 3.信号的处理 3.1信号是什么时候被处…