C++ 泛型编程(二):非类型模板参数,模板特化,模板的分离编译

文章目录

  • 非类型模板参数
  • 函数模板的特化
  • 类模板的特化
    • 全特化
    • 偏特化
      • 部分参数特化
      • 参数修饰特化
  • 模板分离编译
    • 解决方法


非类型模板参数

模板的参数分为两种:

  • 类型参数: 则是我们通常使用的方式,就是在模板的参数列表中在 class 后面加上参数的类型名称。
  • 非类型参数: 则是用一个常量作为模板的参数,在模板中可以当作常量来使用,通常是需要指明大小或者初始化内容的才会用到。

类型参数我们在上一篇博客中讲的很详细了,不再赘述。而非类型参数比较常见的就是 c++ 中的 array
在这里插入图片描述
array 的底层就是直接使用的数组,而数组创建时必须指明大小,并且大小得是个常量,所以就会用到非类型模板参数。

注意:

  1. 浮点数、自定义类型、类对象以及字符串是不允许作为非类型模板参数的。
  2. 非类型参数必须在编译期就能确认结果。

通常情况下非类型模板参数都是使用字符型整型


函数模板的特化

当我们使用模板来实现一个函数,肯定是想利用它来解决逻辑相同但数据类型不同的一些问题,来实现代码的复用,但是也存在某些特例,比如针对某一情景或者某一类型,这个模板需要有特殊的处理,这个时候就需要用到模板的特化。

例如:

template<class T>
bool IsEqual(T str1, T str2)
{return str1 == str2;
}int main()
{char str1[] = "hello";char str2[] = "hello";if (IsEqual(str1, str2))cout << "true";elsecout << "false";
}

在这里插入图片描述

这里不同的原因是传递过去的是两个 char* 类型,他们两个比较的不是字符串的内容,而是指针的地址,这里 str1、str2 是在栈上开辟一块空间后再将 hello 拷贝过去,而 IsEqual 比较的是两者指向的两块不同内存(也就是两个 hello 的内存)的首地址,不可能相同。

如果要比较 char* ,就得用到 strcmp 来对这个情况进行特殊处理, 也就是模板的特化

template<>
bool IsEqual<char*>(char* str1, char* str2)
{return strcmp(str1, str2) == 0;	
}
// extern int strcmp(const char *s1,const char *s2);

在这里插入图片描述
函数模板的特化步骤:

  • 必须要先有一个基础的函数模板
  • 关键字 template 后面接一对 空的尖括号 <>
  • 函数名后面的 <> 中指定需要特化的类型
  • 特化的函数其形参 一定要与 模板的形参 类型完全相同。

类模板的特化

类也是同理,如果需要有特殊情景也需要特化处理:

类模板如下:

template<class T1, class T2>
class test
{
public:test(){cout << "test<T1, T2>" << endl;}private:T1 _x;T2 _y;
};

全特化

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

这里对 test<int,double> 版本特化。

template<>
class test<int, double>
{
public:test(){cout << "test<int, double>" << endl;}private:int _x;double _y;
};int main()
{test<double, double> t1;test<int, double> t2;
}

在这里插入图片描述


偏特化

偏特化即是任何针对模版参数进一步进行条件限制设计的特化版本。

偏特化有两种表现方式:一种是部分参数特化,一种是参数修饰特化

部分参数特化

这里对第二个参数特化,只要第二个参数是 double 就会调用对应特化版本。

template<class T1>
class test<T1, double>
{
public:test(){cout << "test<T1, double>" << endl;}private:T1 _x;double _y;
};int main()
{test<double, double> t1;test<int, double> t2;test<float, double> t3;test<double, int> t4;
}

在这里插入图片描述


参数修饰特化

比如用指针或者引用来修饰类型,也可以进行特化。

template<class T1, class T2>
class test<T1*, T2*>
{
public:test(){cout << "test<T1*, T2*>" << endl;}private:T1* _x;T2* _y;
};int main()
{test<double, int> t1;test<int*, double*> t2;test<float*, double*> t3;test<char*, double> t4;
}	

在这里插入图片描述


模板分离编译

对于一个代码量比较多的项目,通常都会采用声明与定义分离的方法,比如在头文件进行声明,在源文件完成代码的实现,最后通过链接的方法链接成单一的可执行文件。但是 C++ 的编译器却不支持模板的分离编译一旦进行分离编译,就会出现链接错误。

//头文件a.h
template<class T>
bool IsEqual(const T& str1, const T& str2);-------------
//源文件a.cpp
template<class T>
bool IsEqual(const T& str1, const T& str2)
{return str1 == str2;
}
--------------
//test.c
#include<iostream>
#include"a.h"
using namespace std;int main()
{cout << IsEqual(3, 5);cout << IsEqual('a', 'b');
}

这里看上去是没有问题的,但是涉及到了模板的实例化规则。

首先,一个编译单元是指一个 .cpp 文件以及它所 #include 的所有 .h 文件,.h 文件里的代码将会被扩展到包含它的 .cpp 文件里,然后编译器编译该 .cpp 文件为一个 .obj 文件(假定我们的平台是 win32Linux 则是 .o 文件),后者拥有 PE(Portable Executable,即windows可执行文件) 文件格式,并且本身包含的就已经是二进制码,但是不一定能够执行,因为并不保证其中一定有 main函数 。当编译器将一个工程里的所有 .cpp 文件以分离的方式编译完毕后,再由连接器(linker)进行连接成为一个 .exeLinux.out)文件。

那么来看上面的代码,

  • test.ca.cpp 被编译器编译成 test.obja.obj
  • 当编译器处理主函数调用 IsEqual 的时,这是第一次使用模板,会进行实例化,而实例化需要模板的定义,但是在 test.c 中并没有模板的定义,虽然在 test.obj 文件中头文件 a.h 也被展开,但可惜的是 a.h 中只有模板的声明
  • 那么编译器只能寄希望于连接器,希望它能够在其他 .obj 里面找到 IsEqual实例,本题中连接器就是去 a.obj 中找实例,但问题在于,虽然 a.cpp 中有模板的定义,但并没有使用过这个模板,因此 a.obj 中不存在 IsEqual实例。因此连接器只能返回一个连接错误

在这里插入图片描述


解决方法

这个问题其实没有什么完美的解决方法

  • 将声明和定义放到同一个头文件中。(导致头文件代码过于庞大)
  • 模板定义的位置显式实例化。(不实用)

这个问题刘未鹏大佬写的非常好,可以学习一下他的博客

为什么C++编译器不能支持对模板的分离式编译

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

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

相关文章

数据结构 | B树、B+树、B*树

文章目录搜索结构B树B树的插入B树的遍历B树的性能B树B树的插入B树的遍历B*树B*树的插入总结搜索结构 如果我们有大量的数据需要永久存储&#xff0c;就需要存储到硬盘之中。但是硬盘的访问速度远远小于内存&#xff0c;并且由于数据量过大&#xff0c;无法一次性加载到内存中。…

MySQL 索引 :哈希索引、B+树索引、全文索引

文章目录索引引言常见的索引哈希索引自适应哈希索引B树索引聚集索引非聚集索引使用方法联合索引最左前缀匹配规则覆盖索引全文索引使用方法索引 引言 为什么需要索引&#xff1f; 倘若不使用索引&#xff0c;查找数据时&#xff0c;MySQL必须遍历整个表。而表越大&#xff0c;…

服装店怎么引流和吸引顾客 服装店铺收银系统来配合

实体店的同城引流和经营是实体经济的一个重要的一环&#xff0c;今天我们来分享服装行业的实体店铺怎么引流和吸引、留住顾客&#xff0c;并实现复购。大家点个收藏&#xff0c;不然划走就再也找不到了&#xff0c;另外可以点个关注&#xff0c;下次有新的更好的招&#xff0c;…

MySQL 锁的相关知识 | lock与latch、锁的类型、简谈MVCC、锁算法、死锁、锁升级

文章目录lock与latch锁的类型MVCC一致性非锁定读&#xff08;快照读&#xff09;一致性锁定读&#xff08;当前读&#xff09;锁算法死锁锁升级lock与latch 在了解数据库锁之前&#xff0c;首先就要区分开 lock 和 latch。在数据库中&#xff0c;lock 和 latch 虽然都是锁&…

MySQL 存储引擎 | MyISAM 与 InnoDB

文章目录概念innodb引擎的4大特性索引结构InnoDBMyISAM区别表级锁和行级锁概念 MyISAM 是 MySQL 的默认数据库引擎&#xff08;5.5版之前&#xff09;&#xff0c;但因为不支持事务处理而被 InnoDB 替代。 然而事物都是有两面性的&#xff0c;InnoDB 支持事务处理也会带来一些…

MySQL 事务 | ACID、四种隔离级别、并发带来的隔离问题、事务的使用与实现

文章目录事务ACID并发带来的隔离问题幻读&#xff08;虚读&#xff09;不可重复读脏读丢失更新隔离级别Read Uncommitted (读未提交)Read Committed (读已提交)Repeatable Read (可重复读)Serializable (可串行化)事务的使用事务的实现Redoundo事务 事务指逻辑上的一组操作。 …

MySQL 备份与主从复制

文章目录备份主从复制主从复制的作用备份 根据备份方法的不同&#xff0c;备份可划分为以下几种类型&#xff1a; 热备(Hot Backup) &#xff1a; 热备指的是在数据库运行的时候直接备份&#xff0c;并且对正在运行的数据库毫无影响&#xff0c;这种方法在 MySQL 官方手册中又…

C++ 流的操作 | 初识IO类、文件流、string流的使用

文章目录前言IO头文件iostreamfstreamsstream流的使用不能拷贝或对 IO对象 赋值条件状态与 iostate 类型输出缓冲区文件流fstream类型文件模式文件光标函数tellg() / tellp()seekg() / seekp()向文件存储内容/读取文件内容string流istringstreamostringstream前言 我们在使用 …

C++ 右值引用 | 左值、右值、move、移动语义、引用限定符

文章目录C11为什么引入右值&#xff1f;区分左值引用、右值引用move移动语义移动构造函数移动赋值运算符合成的移动操作小结引用限定符规定this是左值or右值引用限定符与重载C11为什么引入右值&#xff1f; C11引入了一个扩展内存的方法——移动而非拷贝&#xff0c;移动较之拷…

且谈关于最近软件测试的面试

前段时间有新的产品需要招人&#xff0c;安排和参加了好几次面试&#xff0c;下面就谈谈具体的面试问题&#xff0c;在面试他人的同时也面试自己。 面试问题是参与面试同事各自设计的&#xff0c;我也不清楚其他同事的题目&#xff0c;就谈谈自己设计的其中2道题。 过去面试总是…

C++ 多态 | 虚函数、抽象类、虚函数表

文章目录多态虚函数重写重定义&#xff08;参数不同&#xff09;协变&#xff08;返回值不同&#xff09;析构函数重写&#xff08;函数名不同&#xff09;final和override重载、重写、重定义抽象类多态的原理虚函数常见问题解析虚函数表多态 一种事物&#xff0c;多种形态。换…

C++ 运算符重载(一) | 输入/输出,相等/不等,复合赋值,下标,自增/自减,成员访问运算符

文章目录输出运算符<<输入运算符>>相等/不等运算符复合赋值运算符下标运算符自增/自减运算符成员访问运算符输出运算符<< 通常情况下&#xff0c;输出运算符的第一个形参是一个 非常量ostream对象的引用 。之所以 ostream 是非常量是因为向流写入内容会改变…

C++ 重载函数调用运算符 | 再探lambda,函数对象,可调用对象

文章目录重载函数调用运算符lambdalambda等价于函数对象lambda等价于类标准库函数对象可调用对象与function可调用对象function函数重载与function重载函数调用运算符 函数调用运算符必须是成员函数。 一个类可以定义多个不同版本的调用运算符&#xff0c;互相之间应该在参数数…

C++ 运算符重载(二) | 类型转换运算符,二义性问题

文章目录类型转换运算符概念避免过度使用类型转换函数解决上述问题的方法转换为 bool显式的类型转换运算符类型转换二义性重载函数与类型转换结合导致的二义性重载运算符与类型转换结合导致的二义性类型转换运算符 概念 类型转换运算符&#xff08;conversion operator&#…

分布式理论:CAP、BASE | 分布式存储与一致性哈希

文章目录分布式理论CAP定理BASE理论分布式存储与一致性哈希简单哈希一致性哈希虚拟节点分布式理论 CAP定理 一致性&#xff08;Consistency&#xff09;&#xff1a; 在分布式系统中的所有数据副本&#xff0c;在同一时刻是否一致&#xff08;所有节点访问同一份最新的数据副…

分布式系统概念 | 分布式事务:2PC、3PC、本地消息表

文章目录分布式事务2PC&#xff08;二阶段提交协议&#xff09;执行流程优缺点3PC&#xff08;三阶段提交协议&#xff09;执行流程优缺点本地消息表&#xff08;异步确保&#xff09;分布式事务 分布式事务就是指事务的参与者、支持事务的服务器、资源服务器以及事务管理器分…

数据结构算法 | 单调栈

文章目录算法概述题目下一个更大的元素 I思路代码下一个更大元素 II思路代码132 模式思路代码接雨水思路算法概述 当题目出现 「找到最近一个比其大的元素」 的字眼时&#xff0c;自然会想到 「单调栈」 。——三叶姐 单调栈以严格递增or递减的规则将无序的数列进行选择性排序…

最长下降子序列

文章目录题目解法DP暴搜思路代码实现贪心二分思路代码实现题目 给出一组数据 nums&#xff0c;求出其最长下降子序列&#xff08;子序列允许不连续&#xff09;的长度。&#xff08;类似于lc的最长递增子序列&#xff09; 示例&#xff1a; 输入&#xff1a; 6 // 数组元素个…

Linux 服务器程序规范、服务器日志、用户、进程间的关系

文章目录服务器程序规范日志rsyslogd 守护进程syslog函数openlog函数setlogmask函数closelog函数用户进程间的关系进程组会话系统资源限制改变工作目录和根目录服务器程序后台化服务器程序规范 Linux 服务器程序一般以后台进程&#xff08;守护进程[daemon]&#xff09;形式运…

IO模型 :阻塞IO、非阻塞IO、信号驱动IO、异步IO、多路复用IO

文章目录IO模型阻塞IO非阻塞IO信号驱动IO多路复用IO异步IOIO模型 根据各自的特性不同&#xff0c;IO模型被分为阻塞IO、非阻塞IO、信号驱动IO、异步IO、多路复用IO五类。 最主要的两个区别就是阻塞与非阻塞&#xff0c;同步与异步。 阻塞与非阻塞 阻塞与非阻塞最主要的区别就…