当前位置: 首页 > web >正文

C++模板进阶使用技巧

  • 非类型模板参数
  • 缺省模板参数
  • 类模板特化
    • 全特化
    • 偏特化
  • 模板的分离编译

我们在前面已经初识了 模板并且在各种数据结构的实现中,熟练掌握了模板的一些基础功能。
至于为什么是基础功能,因为模板还有一些进阶的功能,像非类型模板参数,缺省参数,模板特化之类的。

非类型模板参数

模板是否一定是要接受类型参数呢?能否接受其他类型参数,答案是可以的。
C++的stl中有一种类array其模板为:template < class T, size_t N > class array;
也就是说我们可以这样使用它:

array<int, 10> a1;

这里我们就使用了一个非类型模板参数,还有没有其他非类型模板参数呢?
出乎意料的,答案是没有。也就是说作为非类型模板参数有且仅有size_t.(C++11)

缺省模板参数

事实上,之前实现queue的时候,我们就用过缺省的模板参数,也就是template<class T, class Container = deque<int>>。

类模板特化

对于一个比较类:

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

考虑我们前面实现过的日期类,拿他来比较:

int main()
{Date d1(2025, 3, 12);Date d2(2025, 4, 12);cout << Less(d1, d2) << endl;cout << Less(&d1, &d2) << endl;int x = 1, y = 2;cout << Less(x, y) << endl;cout << Less(&x, &y) << endl;return 0;
}

显然,对于d1,d2比较是可以得到正确的结果,但对于&x,&y就很难得到正确的结果。因为后者比较的是地址大小。
当然我们也可以写一个仿函数来解决这个问题,这里提另一种处理方法,就是对其类模板特化:

template<>
bool Less<Date*>(Date* left, Date* right)//bool Less(Date* left, Date* right)也行
{return *left < *right;
}

实际上类似于函数重载,我们写了一个特殊的函数模板。如果参数匹配这个函数模板就会优先调用这个模板。

事实上我们确实可以通过函数重载来解决这个问题:

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

类模板特化在处理模板函数方面甚至没有写一个重载来的实在。实际上他的主要作用是特化模板类。
此外类模板特化还分为全特化和偏特化。

全特化

对于类:

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

那么运行下面函数:

void TestVector()
{
Data<int, int> d1;
Data<int, char> d2;
}

Output:
Data<T1, T2>
Data<int, char>

偏特化

我们当然可以选择对部分参数特化而非全部参数:

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

这样我们的模板参数T1和T2如果是指针的话就会优先调用这个特化的模板。

  • 但注意我们的T1和T2如果是指针,比如int*。
    那按理说下面这个Data <T1*, T2*>不就应该是Data <int**, int**>
    事实并非如此,而是Data <int*, int*>。
    也就是说T1和T2整体替换成了T1*和 T2*

模板的分离编译

首先我们要了解一个概念,模板的按需实例化

仔细想想,模板实际上是编译器帮我们写函数或者类的一种方式,那么传入参数有无穷多种,编译器自然不可能对所有参数都编译一个代码。

也就是说我们写了一个模板函数之后,编译器是不会直接实例化他的,当你调用这个函数的时候才会。

具体案例:

template<class T,size_t N>
class myarray
{T& operator[](size_t index){size(1);//按需实例化。由于没有调用operator[],即没有实例化,故没有检查出语法错误}size_t size()const{return _size;}size_t _size;
};
int main()
{return 0;
}

可以看到我们的operator[]里面有一个明显的语法错误,就是size的参数传多了。但是这时候编译的话,编译器是不会报错的! 因为operator[]这个函数目前还不存在。
但如果我们在main函数里面调用它的话:

int main()
{myarray a;a[0];
}

这时候编译器就会报错了。

那么接下来我们再看到模板的分离编译。
考虑模板函数的声明定义在不同文件中:

// a.h
namespace myadd
{template<class T>T Add(const T& left, const T& right);
}
// a.cpp
#include"a.h"
template<class T>
T Add(const T& left, const T& right)
{
return left + right;
}

首先这时候由于命名空间的原因,我们的a.cpp实际上实现的不是a.h里面的Add。我们需要给他加上命名空间或者访问限定符:

// a.cpp
namespace myadd
{template<class T>T Add(const T& left, const T& right){return left + right;}
}

这时候看似没有问题了,但如果我们尝试调用他的话:

//main.cpp
#include"a.h"
#include<iostream>
int main()
{int a = 1, b = 10;std::cout << myadd::Add(a, b) << std::endl;return 0;
}

error LNK2019: 无法解析的外部符号 “int __cdecl myadd::Add(int const &,int const &)”
编译器会给我们一个链接错误。
实际上就是上文提及的按需实例化引起的。由于在链接之前,源文件和头文件是不会互通有无的。导致源文件中实现了Add函数的位置,并不知道需要实例化什么类型的函数。这就导致函数无定义。
解决方法有手动实例化:

//a.cpp
#include"a.h"
namespace myadd
{template<class T>T Add(const T& left, const T& right){return left + right;}template int Add(const int& left, const int& right);
}

这里template int Add(const int& left, const int& right);就手动实例化了一个模板参数为int的Add函数。
但这样的方式明显治根不治本。哪有人写了模板函数还得去一个一个手动实例化的?
因此更好的解决方案是:模板函数的定义和声明写在同一个文件里!
不分离就不会有问题了,233.

http://www.xdnf.cn/news/7208.html

相关文章:

  • el-scrollbar 获取滚动条高度 并将滚动条保持在低端
  • mysql数据库故障排查方案
  • 批量处理 Office 文档 高画质提取图片、视频、音频素材助手
  • httpx[http2] 和 httpx 的核心区别及使用场景如下
  • C++ map multimap 容器:赋值、排序、大小与删除操作
  • 【深度学习】残差网络(ResNet)
  • 图书管理系统
  • 滑动窗口算法详解与C++实现
  • 【背包dp】小结
  • 20250518 黎曼在三维空间中总结的一维二维的规律,推广到高维度合适吗?有没有人提出反对意见
  • Power BI Desktop运算符和新建列
  • 职场方法论总结(3)-金字塔原理
  • Redis的持久化机制
  • 深入探索PointNet:点云处理的革命性算法
  • 【MySQL】02.数据库基础
  • 安装和升级到devExpress23.1.7
  • #Redis黑马点评#(七)实战篇完结
  • 2025 ISCC 练武赛Pwn-wp(含附件)
  • 知识图谱(KG)与大语言模型(LLM)
  • 《算法导论(第4版)》阅读笔记:p83-p85
  • buck变换器的simulink/matlab仿真和python参数设计
  • 互联网大厂Java面试:从Spring Boot到微服务架构的技术深挖
  • 进程概念及操作系统的知识点
  • 基于STM32的多传感器融合的设施农业小型搬运机器人避障控制系统设计
  • React方向:react的基本语法-数据渲染
  • 备战!全国青少年信息素养大赛图形化编程-省赛——求最小公倍数
  • 【CF】Day61——Codeforces Round 939 (Div. 2) CD (思维构造 | 思维构造 + dfs枚举)
  • Python实例题:基于scrapy爬虫的天气数据采集
  • 构建 TypoView:一个富文本样式预览工具的全流程记录
  • 基于RDMA的跨节点GPU显存共享技术实践