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

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.

相关文章:

  • NY337NY340美光固态颗粒NC010NC012
  • wsl2中Ubuntu22.04配置静态IP地址
  • 基于STM32F103与Marvell88W8686的WIFI无线监控视频传输系统研发(论文)
  • 1.5 MouseDown,MouseUp,LostMouseCapture的先后顺序
  • 三、高级攻击工具与框架
  • OpenHarmony SIM卡信号值整体流程分析
  • 【Vue篇】数据秘语:从watch源码看响应式宇宙的蝴蝶效应
  • 仿腾讯会议——退出房间
  • spark数据处理练习题详解【上】
  • STM32 OTA 中断向量表重定向
  • Node.js 框架
  • 数组-长度最小的子数组
  • USB接口介绍
  • dijkstra算法加训上 之 分层图最短路
  • HashMap的扩容机制
  • AM32电调学习解读五:tenKhzRoutine
  • 二十、案例特训专题3【系统设计篇】web架构设计
  • nginx相关面试题30道
  • 【嵙大o】C++作业合集
  • 【Linux】利用多路转接epoll机制、ET模式,基于Reactor设计模式实现
  • 央媒:设施老化、应急预案套模板,养老机构消防隐患亟待排查
  • 江南考古文脉探寻
  • 上海博物馆展览进校园,“小先生”传递文物知识
  • 雷军内部演讲回应质疑:在不服输、打不倒方面,没人比我们更有耐心
  • 机构发布“2025中国高职院校排名”
  • 牛市早报|4月新增社融1.16万亿,降准今日正式落地