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

【C++】非类型的模板参数,特化

目录

1.类型模板参数和非类型模板参数

2.特化

3. 模板的分离编译

4.模板的优缺点


1.类型模板参数和非类型模板参数

之前写模板传的都是类型——类型模板参数

现在想定义两个静态数组,数组长度不同,就可以用模板参数传数值而不是传类型

非类型模板参数——是一个常量,只适用于整型(int,short,char,size_t...)

那么这个静态数组可以用array,是一个固定大小的顺序表

 

其实可以发现他的模板参数就是有类型的也有非类型的

成员函数没有头尾插,有迭代器和范围for,他对比的是C的静态数组,并且他的数据也存在栈上

 

 为什么可以兼容C还要搞出他?

他们对越界的检查不一样

C对越界读不检查,越界写抽查

C++的array越界读写都能查,因为重载了[ ],operator对其检查


2.特化

模板的特化,使用模板可以实现一些与类型无关的代码,但对于一些特殊的类型可能会得到错误结果

按照类型分类

  • 函数模板特化

比如之前我们实现的Less,如果传的是类型指针就会比较地址,而不是解引用之后的值

 我们的解决方案是自己写一个仿函数

但是有一点点挫

现在我们可以进行特化

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

一定要注意,函数模板特化,一定要有函数模板,然后才是针对某些类型的特殊处理(用函数模板特化) ,只有特化是会报错的,因为一般类型没有处理

 注意参数列表的统一,说的通俗一些,特化就是特殊处理,但是本质不应该改变

第一个Less是对引用的比较,但是第二个Less突然变成了对“值”的比较

 应该写成

 

 当然对于函数模板特化,这个类型匹配非常恶心,建议还是直接写成重载

  • 类模板特化 

和刚才的规则一样

假设我们对A类的<int,int> 类型特化

template<class T1, class T2>
class A
{
public:A(){cout << "A<T1, T2>" << endl;}
private:T1 _a1;T2 _a2;
};template<>
class A<int, int>
{
public:A(){cout << "A<int, int>" << endl;}
private:int _a1;int _a2;
};

调用结果: 

 


按照特化程度分类: 

  • 全特化

 全部参数特化

还是上面那个例子,只对<int,int>类型特化,<int,X>不可以,<X,int>也不可以

 

  • 偏特化 /半特化

 部分参数特化,是对参数类型的进一步限制

只要类型是<X,char>就匹配

template<class T1, class T2>
class A
{
public:A(){cout << "A<T1, T2>" << endl;}
private:T1 _a1;T2 _a2;
};
template<class T>
class A<T, char>
{
public:A(){cout << "A<int, int>" << endl;}
private:T _a1;char _a2;
};

 优先级总结 

 a1可以三种都可以匹配,但是他选择全特化

a2可以匹配半特化和类模板,他选择前者

最后一个没有选择


特化本质:编译器的参数匹配原则


3. 模板的分离编译

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

 

无法解析的外部符号就是链接错误,因为在编译的时候会生成符号表,把语言级别的代码变成汇编语言,然后汇编形成可重定位的目标二进制文件(.obj),最后链接自己形成的几个.obj然后生成可执行程序

在函数定义中,无法确定T的类型,故没有实例化

在函数模板调用中,没有函数实现,还是没办法实例化

因此最后这个函数模板没有实例化,无法生成符号,链接时自然找不到

解决方法:声明和定义都写在.h/.hpp中 

 4.模板的优缺点

  • 优点:模板复用了代码,节约资源,更快迭代开发,增强代码灵活性
  • 缺点:导致代码膨胀,编译时间变长,出现模板编译错误时,报错信息混乱,不易定位错误

 

http://www.lryc.cn/news/38594.html

相关文章:

  • 核方法(kernel Method)
  • 消息队列MQ用来做什么的,市场上主流的四大MQ如何选择?RabbitMQ带你HelloWorld!
  • 2023年中国高校计算机大赛-团队程序设计天梯赛(GPLT)上海理工大学校内选拔赛(同步赛) A — E
  • 一文分析Linux v4l2框架
  • MFC常用控件使用(文本框、编辑框、下拉框、列表控件、树控件)
  • 13 node 程序后台执行加上 tail 命令, 中断 tail 命令, 同时也中断了 node 程序
  • 52癫痫发作预测的有效双自注意力残差网络
  • 【计算机网络】Tcp IP 面试题相关
  • 【MySQL】MySQL的存储引擎
  • es6动态模块import()
  • 【Flask】Jinja2模板(十四)
  • Mr. Cappuccino的第49杯咖啡——冒泡APP(升级版)之基于Docker部署Gitlab
  • 《机器学习》基础概念之【P问题】与【NP问题】
  • WinRAR安装教程
  • C++:vector和list的迭代器区别和常见迭代器失效问题
  • SpringSecurity如何实现前后端分离
  • 为ubuntu 18.04添加蓝牙驱动
  • Stable Diffusion Prompt用法
  • jenkins问题
  • 阅读笔记DeepAR: Probabilistic Forecasting with Autoregressive Recurrent Networks
  • 01.Java的安装
  • 【C语言深度剖析】关键字(全)
  • English Learning - L2 语音作业打卡 双元音 [aʊ] [əʊ] Day15 2023.3.7 周二
  • 记第一次面试的过程(C++)
  • 06 电力电子仿真 MATLAB/Simulink
  • 搞懂面向对象这五大概念,才算真正跨过初学者到开发者的“分水岭“
  • 基于DelayQueue实现的延时队列
  • MATLAB实现层次分析法AHP及案例分析
  • Vue 3.0 TypeScript支持
  • STM8S系列基于IAR标准外设printf输出demo