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

【C++干货铺】非类型模板 | 模板特化 | 模板分离编译

=========================================================================

个人主页点击直达:小白不是程序媛

C++系列专栏:C++干货铺

代码仓库:Gitee

=========================================================================

目录

非类型模板参数

模板的特化

什么是模板特化?

函数模板特化

类模板的特化

全特化

偏特化

模板的分离编译

什么是分离编译?

模板的分离编译

模板总结


非类型模板参数

模板参数分类类型形参非类型形参

类型形参即:

出现在模板参数列表中,跟在class或者typename之类的参数类型名称。

非类型形参:

就是用一个常量作为类(函数)模板的一个参数,在类(函数)模板中可将该参数当成常量来使用。 

template < class T ,size_t N >
class Array
{
private:T _a[N];
};
int main()
{Array<int, 10> a1;Array<double, 100> a2;return 0;
}

注意:

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


模板的特化

什么是模板特化?

通常情况下,使用模板可以实现一些与类型无关的代码,但对于一些特殊类型的可能会得到一些错误的结果,需要特殊处理,比如:实现了一个专门用来进行小于比较的函数模板

template <class T>
bool Less(T& left, T& right)
{return left < right;
}
int main()
{int a = 2;int b = 1;cout << Less(a,b) << endl;int* pa = &a, * pb = &b;cout << Less(pa, pb) << endl;return 0;
}

可以看到,Less绝对多数情况下都可以正常比较,但是在特殊场景下就得到错误的结果。上述示例中,pa指向的a显然大于pb指向的b,但是Less内部并并没有比较pa和pb指向的内容,而是比较pa和pb指针的地址,这就无法达到预期的而错误。

此时,就需要对模板进行特化。

即:在原模板类的基础上,针对特殊类型所进行特殊化的实现方式。模板特化中分为函数模板特化与类模板特化。 

函数模板特化

函数模板的特化步骤

1. 必须要先有一个基础的函数模板
2. 关键字template后面接一对空的尖括号<>
3. 函数名后跟一对尖括号,尖括号中指定需要特化的类型
4. 函数形参表: 必须要和模板函数的基础参数类型完全相同,如果不同编译器可能会报一些奇怪的错误。

//模板的特化
template <class T>
bool Less(T left, T right)
{return left < right;
}
template<>
bool Less<int*>(int*left,int*right)
{return *left < *right;
}
int main()
{int a = 2;int b = 1;cout << Less(a,b) << endl;int* pa = &a, * pb = &b;cout << Less(pa, pb) << endl;return 0;
}

注意:

一般情况下如果函数模板遇到不能处理或者处理有误的类型,为了实现简单通常都是将该函数直接给出。 

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

该种实现简单明了,代码的可读性高,容易书写,因为对于一些参数类型复杂的函数模板,特化时特别给出,因此函数模板不建议特化。

类模板的特化

全特化

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

//类特化template <class T1,class T2>
class Test
{
public:Test(){cout << "Test<T1,T2>" << endl;}
private:T1 _a;T2 _b;
};
template <>
class Test<int,char>
{
public:Test(){cout << "Test<int,char>" << endl;}
private:int _a;char _b;
};
int main()
{Test<int, int> t1;Test<int, char> t2;return 0;
}

偏特化

偏特化:

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

比如对于以下类模板:

//偏特化
template <class T1,class T2>
class Test
{
public:Test(){cout << "Test<T1,T2>" << endl;}
private:T1 _a;T2 _b;
};

偏特化有以下两种表现方式:

部分特化

  • 将模板参数类表中的一部分参数特化。 
template <class T>
class Test<T, int>
{
public:Test(){cout << "Test<T,int>" << endl;}
private:T _a;int _b;
};
  • 参数更进一步的限制

偏特化并不仅仅是指特化部分参数,而是针对模板参数更进一步的条件限制所设计出来的一个特化版本。 

//全限制
template<class T1,class T2>
class Test< T1*,  T2*>
{
public:Test(){cout << "Test<T1*,T2*>" << endl;}
};

模板的分离编译


什么是分离编译?

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

模板的分离编译

假如有以下场景,模板的声明与定义分离开,在头文件中进行声明,源文件中完成定义:

//Add.h
template<class T>
T Add(const T& left, const T& right);
//Add.cpp
template<class T>
T Add(const T& left, const T& right)
{
return left + right;
}
// main.cpp
#include"a.h"
int main()
{
Add(1, 2);
Add(1.0, 2.0);
return 0;
}

 分析:

解决办法:

1. 将声明和定义放到一个文件 "xxx.hpp" 里面或者xxx.h其实也是可以的。推荐使用这种。
2. 模板定义的位置显式实例化。这种方法不实用,不推荐使用。


模板总结

优点:

1. 模板复用了代码,节省资源,更快的迭代开发,C++的标准模板库(STL)因此而产生
2. 增强了代码的灵活性 

缺点:

1. 模板会导致代码膨胀问题,也会导致编译时间变长
2. 出现模板编译错误时,错误信息非常凌乱,不易定位错误


相关文章:

  • 【C4D如何将多个选集设置为一个选集】
  • 前端(HTML + CSS + JS)
  • 成品日志函数
  • Scrapy爬虫异步框架(一篇文章齐全)
  • 插入排序(形象类比)
  • ubuntu修改系统语言
  • Windows系统管理之备份与恢复
  • PgSQL技术内幕-Analyze做的那些事-pg_stat_all_tables
  • Hibernate 脏检查和刷新缓存机制
  • 【开源】基于Vue.js的天然气工程运维系统的设计和实现
  • EMG肌肉信号处理合集 (一)
  • 【DP】mobiusp正在创作乐曲
  • ubuntu20.04配置OpenCV的C++环境
  • 深度学习之基于YoloV3杂草识别系统
  • GIT | 基础操作 | 初始化 | 添加文件 | 修改文件 | 版本回退 | 撤销修改 | 删除文件
  • 2017 前端面试准备 - 收藏集 - 掘金
  • Akka系列(七):Actor持久化之Akka persistence
  • ESLint简单操作
  • ES学习笔记(10)--ES6中的函数和数组补漏
  • MaxCompute访问TableStore(OTS) 数据
  • Phpstorm怎样批量删除空行?
  • Theano - 导数
  • Unix命令
  • 得到一个数组中任意X个元素的所有组合 即C(n,m)
  • 基于 Babel 的 npm 包最小化设置
  • 精益 React 学习指南 (Lean React)- 1.5 React 与 DOM
  • 我的zsh配置, 2019最新方案
  • 我这样减少了26.5M Java内存!
  • 找一份好的前端工作,起点很重要
  • 专访Pony.ai 楼天城:自动驾驶已经走过了“从0到1”,“规模”是行业的分水岭| 自动驾驶这十年 ...
  • 资深实践篇 | 基于Kubernetes 1.61的Kubernetes Scheduler 调度详解 ...
  • ​iOS实时查看App运行日志
  • ​批处理文件中的errorlevel用法
  • # Panda3d 碰撞检测系统介绍
  • (1) caustics\
  • (1)STL算法之遍历容器
  • (3)选择元素——(14)接触DOM元素(Accessing DOM elements)
  • (C++)栈的链式存储结构(出栈、入栈、判空、遍历、销毁)(数据结构与算法)
  • (echarts)echarts使用时重新加载数据之前的数据存留在图上的问题
  • (env: Windows,mp,1.06.2308310; lib: 3.2.4) uniapp微信小程序
  • (HAL库版)freeRTOS移植STMF103
  • (Note)C++中的继承方式
  • .Net - 类的介绍
  • .NET Core IdentityServer4实战-开篇介绍与规划
  • .NET Core 控制台程序读 appsettings.json 、注依赖、配日志、设 IOptions
  • .NET(C#、VB)APP开发——Smobiler平台控件介绍:Bluetooth组件
  • @Async注解的坑,小心
  • @RequestMapping 的作用是什么?
  • @RunWith注解作用
  • [ vulhub漏洞复现篇 ] ECShop 2.x / 3.x SQL注入/远程执行代码漏洞 xianzhi-2017-02-82239600
  • [<死锁专题>]
  • [BZOJ 3680]吊打XXX(模拟退火)
  • [C++]Leetcode17电话号码的字母组合
  • [C++]类和对象(中)
  • [C语言]编译和链接