C++语法---模板进阶知识

绪论​
“那些看似不起波澜的日复一日,会在某天让你看到坚持的意义。”本篇文章主要写到非类型的模板参数、模板的特化、模板的分离编译问题、以及适配器和仿函数的使用讲解,在之前已经将模板的基本使用进行了学习(可见c++模板)话不多说安全带系好,发车啦(建议电脑观看)。

思维导图:

附:红色,部分为重点部分;蓝颜色为需要记忆的部分(不是死记硬背哈,多敲);黑色加粗或者其余颜色为次重点;黑色为描述需要


1.非类型模板参数

在模板参数中分为了类型模板参数和非类型模板参数

类型模板参数:出现在模板中在class/typename后的参数类型名称
非类型模板参数用一个常量作为类的一个模板参数,在类中就能直接使用这个常量

例如,在一个类中成员变量为一个静态数组时,我们可以通过这个常量来控制该数组的大小:

template<class T,size_t N>
class Aarry
{
private:T base[N];
};int main()
{Aarry<int, 5> a;return 0;
}

在这里插入图片描述
注意的是:

  1. 该非类型的模板参数支持整形(浮点型不可以,在c++11及以前暂时不能使用)
  2. 且只能是常量,而不能为变量

2.模板的特化

模板的特化分为两种:

  1. 函数模板特化
  2. 类模板特化

使用方法不同通过具体实例来看:

2.1 对于类的特化:

template<class T , class Z>
class A
{
public:A() { cout << "class A<T,Z>" << endl; }private:T _a;Z _z;
};//注意下方!!!!
//此处就是一个类的**全特化**(所有模板参数都进行了特化)
//方法如下就是在类旁边加上特化的类型
template<>
class A<int,double>
{
public:A() { cout << "class A<int,double>" << endl; }private:int _a;double _z;
};template<class T1>
class C<T1,double>
{
public:C() { cout << "class C<T1,double>" << endl; }private:T1 _a;double _z;
};int main()
{A<int,double> a1;//调用特化的类A<int,int> a1;//调用特化的类A<int*, int*> a2;//正常实例化对象return 0;
}

运行结果为:在这里插入图片描述
类模板的特化(特化的前提是得 有原模板)来说分为:

  1. 全特化:
    从上方也能看出方法为:
    在这里插入图片描述
    也就是将特化的类型写在类名旁边进行特化操作
  2. 偏特化
    特化一部分初参数类型,还剩一部分类型仍然为模板参数
    方法为:
    在这里插入图片描述
    同样也就是将特化的类型写在类名旁边进行特化操作,而对于不想特化的就仍然写成模板参数

注意如果有歧义那么编译器会找最匹配的(如全特化和偏特化同时有部分特化相等,当全特化更加匹配时那么就会走全特化,具体如下面这种情况那么就会走到全特化
在这里插入图片描述
此处上面的例子就是全特化更加匹配:
全特化在这里插入图片描述
在这里插入图片描述

偏特化的特别使用,对参数的限制

  1. 此处当T1、T2是指针类型时会走该特化类:
template <class T1,class T2>
class Date<T1*,T2*>//此处当T1、T2是指针时走这里!!
{
public:Date() { cout << "Data<T1*, T2*>" << endl; }
private:T1 _d1;T2 _d2;
};

在这里插入图片描述

  1. 此处当T1、T2是引用类型时会走该特化类:
template <class T1, class T2>
class Date<T1&, T2&>
{
public:Date() { cout << "Data<T1&, T2&>" << endl; }
private:T1 _d1;T2 _d2;
};

在这里插入图片描述


2.2函数模板的特化

对于函数的特化见于下面例子:

template<class T>
bool Less(T left, T right)
{return left < right;
}int main()
{Date* d1 = new Date(2000, 10, 1);Date* d2 = new Date(2000, 10, 2);cout << Less(d1, d2) << endl;
//预计为日期d1 < 日期d2 返回打印1return 0;
}

上面程序结果为:
在这里插入图片描述
在这里插入图片描述

我们预期的结果是1(日期d1 < 日期d2),可发现这个结果是不确定的(可能1/0),分析发现这是因为Less函数接收到的T类型为Date*(指针类型)而d1、d2的地址每次分配都不一样 可大可小,对此这里我们应该是要取到*d1 与 *d2 进行比较而不是d1 与 d2 比,所以我们应该对这种情况进行特殊化处理(函数特化)具体如下:

//特化方法如下:
template<>
bool Less<Date*>(Date* left, Date* right)//当接收到Date*类型时进行特殊化处理
{return *left < *right;//去取日期来比较 而非地址
}

通过上面能看出函数特化的写法:

在这里插入图片描述
在有原函数模板的前提下,特化一个函数的方法:
1. 写下template<> ,2. 再将特化的类型用尖括号(<>)写到函数名旁边,3. 并且还要改变函数内所用到该类型的类型名

注意的特殊点
在这里插入图片描述
也就是此处的const是要修饰变量本身left、right(&left、&right),而不是*left、right 所以把const挪到后面
理解如同: const int * p (修饰指针
p)与 int * const p(修饰变量p)

类模板用特化,函数模板不用特化用重载(编译器一般会用最适配的,故有更匹配的重载时,会直接调用该函数)!

3.模板的分离编译问题

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

具体:当对有模板的函数进行分离编译时会出现下面问题:

//在 Add.cpp 的函数实现
template<class T>
T Add(const T& t1, const T& t2)
{return t1 + t2;
}
//在 Add.h 头文件中的函数声明
template<class T>
T Add(const T& t1, const T& t2);//类函数的分离编译//在 main.cpp 中
int main()
{cout << Add(1,2) << endl;return 0;
}

在这里插入图片描述
反观普通函数就能正常跑!

//fun.cpp
void fun()
{cout << "void fun()" << endl;
}
//fun.h
void fun();//普通函数的分离编译//main.cpp
int main()
{fun();//cout << Add(1,2) << endl;return 0;
}

在这里插入图片描述
对此发现找不到Add函数主要是因为模板的原因,那为什么会加上模板就不行了嘞??
编译器编译的步骤是:

  1. 预处理(展开头文件、替换宏、条件编译) test.i
  2. 编译(检查语法、生成汇编)test.s
  3. 汇编(将汇编代码转换为二进制机器码)test.o
  4. 链接 (将多个.o的目标文件合并)a.out

此处主要是因为对于模板函数来说,他在运行阶段时才会实例化产生地址。因此在编译阶段始终都是没有地址的,而编译时会给Add函数变成汇编指令(call Add(地址?)),一般分离编译的函数在链接时就会通过合并符号表(存放着符号汇总(汇总一些全局变量和函数)的符号)将此处的地址补齐(此处具体可见博客),而模板函数因为无实际地址就无法进行补齐。

解决方案:

  1. 显示实例化:意思就如名字一样进行对应所需的实例化告诉编译器类型如下:在这里插入图片描述但缺点也很明显,当类型变成其他时就需要再写一个 (template double Add < double > … )
  2. 把分离编译的模板函数的实现和声明一起写在同一个头文件里(有些地方会把.h 改成 .hpp),这样当预处理时就会把实现也带进main.cpp中当Add编译时就会找到实现,那么就会直接进行实例化生成地址,不再依赖于链接

4.模板的缺点

  1. 导致代码膨胀问题,同时会导致编译时间变长
  2. 模板出现编译错误时,错误信息非常难看(方法:先看第一个错误),定位不准

5.适配器

适配器是一种设计模式,常用于一些可以套用的结构中,如stack、queue他们底层都需要一个存储数据的容器,而这个容器就能通过适配器来直接调用的来使用,不需要再自行实现。如stack他的适配器就能是vector、list、dequeue。
而适配器的位置常常是写在模板参数处的
在这里插入图片描述
基本的使用方法:创建对象后直接使用该对象即可!
在这里插入图片描述
如:插入数据
在这里插入图片描述

6.仿函数

仿函数:一种类似于函数的类
仿函数的创建,创建一个类,然后在类中重载operator()
仿函数的使用:通过对象调用operator() ,达到模仿函数的样子去使用(本质还是调用了类中的运算符重载函数)。
在这里插入图片描述

  1. 构建成对象,通过实例化对象后用对象来使用(见上 图标注1),
    1.1传递给某写函数当函数参数(类似于函数指针!)(如下改变sort的第三个参数就会有不同的效果,实际上把对象传过去在该函数中使用这个仿函数)在这里插入图片描述less排升序(此处模拟实现less(Less))在这里插入图片描述greater排降序
    在这里插入图片描述
    2.当成模板参数传给某类,在类内部通过实例化对象后再直接使用
    具体如下:
    在这里插入图片描述在这里插入图片描述

总结来说仿函数就是一个类似于函数的类,我们只需要将其实例化对象后==通过对象()==就能使用这个仿函数

本章完。预知后事如何,暂听下回分解。

如果有任何问题欢迎讨论哈!

如果觉得这篇文章对你有所帮助的话点点赞吧!

持续更新大量C++细致内容,早关注不迷路,我们下章再会。

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

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

相关文章

Linux如何修改主机名(hostname)(亲测可用)

文章目录 背景Linux如何修改主机名&#xff08;hostname&#xff09;方法方法1. 使用 hostnamectl 命令示例 2. 编辑 /etc/hostname 文件注意事项 背景 我创建虚拟机的时候没设置主机名&#xff0c;现在显示localhost&#xff0c;有点尴尬&#x1f605;&#xff1a; 需要重新设…

你一定要学会的Java语法 -- 【继承】

书接上回&#xff0c;我们已经学完了类和对象&#xff0c;今天内容可能有一点难&#xff0c;相信自己能跨过这道坎。 目录 一. 继承 1.什么是继承 2. 继承的概念 3. 继承的语法 4.父类成员访问 子类和父类成员变量同名 子类和父类成员方法同名 5.super关键字 6.子类构…

实操创建属于自己的亚马逊云科技VPS服务:Amazon Lightsail

本文主要讲述如何独立创建自己的亚马逊云科技VPS服务&#xff0c;希望此文能帮助你对亚马逊云科技VPS服务也就是Amazon Lightsail&#xff0c;有个新的认识&#xff0c;对所使用的VPS有所帮助。 Amazon Lightsail是一款无论云计算的新手还是专家&#xff0c;都可通过其快速启动…

MSF图形化工具Viper快速安装

简介 Viper(炫彩蛇)是一款图形化内网渗透工具,将内网渗透过程中常用的战术及技术进行模块化及武器化. Viper(炫彩蛇)集成杀软绕过,内网隧道,文件管理,命令行等基础功能. Viper(炫彩蛇)当前已集成70个模块,覆盖初始访问/持久化/权限提升/防御绕过/凭证访问/信息收集/横向移动等…

Java图像编程之:Graphics

一、概念介绍 1、Java图像编程的核心类 Java图像编程的核心类包括&#xff1a; BufferedImage&#xff1a;用于表示图像的类&#xff0c;可以进行像素级的操作。Image&#xff1a;表示图像的抽象类&#xff0c;是所有图像类的基类。ImageIcon&#xff1a;用于显示图像的类&a…

网络爬虫代理ip有什么好处?爬虫工作使用代理IP有哪些优势?

在爬虫工作中&#xff0c;使用代理IP有很多好处&#xff0c;可以帮助爬虫程序更加高效地完成任务。以下是使用代理IP的几个优势&#xff1a; 1. 增加匿名性 使用代理IP可以隐藏爬虫程序的真正IP地址&#xff0c;增加匿名性&#xff0c;避免被目标网站封禁。通过代理IP&#xff…

ida81输入密码验证算法分析以及破解思路

本文分析了ida81对输入密码的验证流程&#xff0c;分别对输入密码到生成解密密钥、密码素材的生成过程以及文件数据的加密过程这三个流程进行分析&#xff0c;并尝试找一些可利用的破绽。很遗憾&#xff0c;由于水平有限&#xff0c;目前也只是有个思路未能完全实现&#xff0c…

【中间件篇-Redis缓存数据库02】Redis高级特性和应用(慢查询、Pipeline、事务、Lua)

Redis高级特性和应用(慢查询、Pipeline、事务、Lua) Redis的慢查询 许多存储系统&#xff08;例如 MySQL)提供慢查询日志帮助开发和运维人员定位系统存在的慢操作。所谓慢查询日志就是系统在命令执行前后计算每条命令的执行时间&#xff0c;当超过预设阀值,就将这条命令的相关…

微信聊天,收到二维码图片就自动帮你提取出来的方法

10-3 如果你是二维码收集的重度用户&#xff0c;那我非常推荐你好好阅读本文&#xff0c;也许可以帮你解决你的问题&#xff0c;比如做网推的人&#xff0c;需要常年混迹在各种微信群&#xff0c;那如何在各个微信群中收集到群友分享出来的二维码&#xff0c;并且要立即保存出…

故障诊断模型 | Maltab实现ELM极限学习机的故障诊断

文章目录 效果一览文章概述模型描述源码设计参考资料效果一览 文章概述 故障诊断模型 | Maltab实现ELM极限学习机的故障诊断 模型描述 在机器学习领域,我们常常需要通过训练数据来学习一个函数模型,以便在未知的数据上进行预测或分类。传统的神经网络模型需要大量的参数调整和…

计算机毕业设计选题推荐-校园交流平台微信小程序/安卓APP-项目实战

✨作者主页&#xff1a;IT研究室✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Python…

9.spark自适应查询-AQE之动态调整Join策略

目录 概述动态调整Join策略原理实战 动态优化倾斜的 Join原理实战 概述 broadcast hash join 类似于 Spark 共享变量中的广播变量&#xff0c;Spark join 如果能采取这种策略&#xff0c;那join 的性能是最好的 自适应查询AQE(Adaptive Query Execution) 动态调整Join策略 原…