C++初阶 —— 模板进阶

2023-05-16

目录

一,非类型模板参数

模板参数分类

二,模板特化

函数模板特化

类模板特化

三,模板分离编译

分离编译

链接失败原因

解决方法

模板优点

模板缺点


一,非类型模板参数

模板参数分类

  • 类型形参,模板参数列表中,在class/typename后的名称;
  • 非类型形参,使用一个常量作为类或函数模板的一个参数,可将此参数当成常量使用;
template<class T, size_t N=10>
class array
{
public:
	size_t size()
	{
		return sizeof(_array) / sizeof(T);
	}	
private:
	T _array[N];
};

int main()
{
	array<char>arr;
	std::cout << arr.size() << std::endl;
	return 0;
}

注:

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

二,模板特化

        通常,使用模板可实现与类型无关的代码,但对一些特性的类型可能会得到错误的结果,此时需对模板进行特化,即在原模板类的基础上,针对特殊类型进行特殊化实现方式;

template<class T>
bool IsEqual(const T& left, const T& right)
{
	return left == right;
}

int main()
{
    //实参为整数,调用IsEqual没问题
	cout << IsEqual(1, 1) << endl;

    //实参为指针,调用IsEqual实际上是指针的比较
	const char p1[] = "ab";
	const char p2[] = "ab";
	cout << IsEqual(p1, p2) << endl;
	return 0;
}
template<class T>
bool IsEqual(const T& left, const T& right)
{
	return left == right;
}

//优先匹配现成的
bool IsEqual(const char* left, const char* right)
{
	return strcmp(left, right) == 0;
}

int main()
{
	cout << IsEqual(1, 1) << endl;

	const char p1[] = "ab";
	const char p2[] = "ab";
	cout << IsEqual(p1, p2) << endl;
	return 0;
}

函数模板特化

  • 必须先有一个基础的函数模板;
  • 关键字template后面接<>;
  • 函数名后需接<>,其内指定特化类型;
  • 函数形参列表,必须要和模板函数的基础参数类型完全相同;
//模板特化
template<>
bool IsEqual<const char* const >(const char* const& left, const char* const& right)
{
	return strcmp(left, right) == 0;
}

//如函数模板遇到不能处理或处理有误的类型,可直接将该函数给出
bool IsEqual(char* left, char* right)
{
	return strcmp(left, right) == 0;
}

类模板特化

  • 全特化,即将模板参数列表中所有参数都确定化;
  • 偏特化或半特化,部分参数特化、或参数更进一步限制;
//类模板
template<class T1, class T2>
class Data
{
public:
	Data()
	{
		cout << "Data<T1, T2>" << endl;
	}
private:
	T1 _d1;
	T2 _d2;
};
//类模板特化,全特化
template<>
class Data<int, double>
{
public:
	Data()
	{
		cout << "Data<int, double>" << endl;
	}
private:
	int _d1;
	double _d2;
};
//类模板偏特化
template<class T1>
class Data<T1, double>
{
public:
	Data()
	{
		cout << "Data<T1, double>" << endl;
	}
private:
	T1 _d1;
	double _d2;
};
//参数偏特化为指针类型
template<class T1, class T2>
class Data<T1*, T2*>
{
public:
	Data()
	{
		cout << "Data<T1*, T2*>" << endl;
	}
private:
	T1 _d1;
	T2 _d2;
};
//参数偏特化为引用类型
template<class T1, class T2>
class Data<T1&, T2&>
{
public:
	Data()
	{
		cout << "Data<T1&, T2&>" << endl;
	}
private:
	T1 _d1;
	T2 _d2;
};

三,模板分离编译

程序在计算机中的执行过程:

分离编译

  • 一个程序或项目由若干个源文件共同实现,每个源文件单独编译生成目标文件,最后将所有目标文件链接生成单一的可执行文件的过程;

模板不可分离编译,即在头文件声明,源文件定义,此时会报链接错误;

//头文件add.h
template<class T>
T add(const T& a, const T& b);

//源文件add.cpp
template<class T>
T add(const T& a, const T& b)
{
	return a + b;
}

//源文件main.cpp
#include "head.h"
#include<iostream>
int main()
{
	std::cout << add(1, 2) << std::endl;;
	return 0;
}

链接失败原因

  • 源文件add.cpp/main.cpp,分别独立编译最后生成add.obj/main.obj,而在main.cpp中并没有add函数的实现,所以main.obj中仅有一条call指令(声明调用add函数,此时无实际调用地址),将add当成外部链处理(即认为此函数实现代码在另一个.obj文件中);
  • 在链接时,指定call指令后调用add的地址,来实现调用;模板函数是不能直接编译成代码,需实例化才可,但add.cpp中,并没有用到模板函数,也就没有实例化(C++标准规定),也就没有add实现地址;

注:在分离编译环境下,源文件都是独立编译的,编译器并不知道其他源文件的存在,对于函数的调用只能靠链接器;在普通函数情况下没有问题,但遇到模板时就会容易导致链接错误,因为模板仅在需要时才会实例化;当编译器遇到模板声明时,不会实例化模板仅常见具有外部链接的符号期待链接时能够得到符号的地址;

解决方法

  • 将声明和定义放在同一个.h(.hpp)文件中;
  • 在模板定义的文件中,显示实例化,此方法不推荐;
//头文件add.h
template<class T>
T add(const T& a, const T& b);

template<class T>
T add(const T& a, const T& b)
{
	return a + b;
}
//源文件add.cpp
template<class T>
T add(const T& a, const T& b)
{
	return a + b;
}
//显示实例化
//缺点是用一个类型就需实例化一个类型
template
int add(const int& a, const int& b);

//或
template<>
int add(const int& a, const int& b)
{
	return a + b;
}

注:模板是按需实例化的,没有实例化编译器不会检测模板内部语法错误,对类模板的成员函数也是按需实例化的,即调用时才实例化;

模板优点

  • 模板复用了代码,节省了资源;
  • 更快的迭代开发,C++模板标准模板库STL因此产生;
  • 增加了代码的灵活性;

模板缺点

  • 模板会导致代码膨胀问题,编译会过长;
  • 出现模板编译错误时,错误信息凌乱,不易定位错误;

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C++初阶 —— 模板进阶 的相关文章

  • Springboot+Netty搭建TCP客户端-多客户端

    之前搭建了一个Springboot 43 Netty服务端的应用 xff0c 既然有服务端 xff0c 自然也有客户端的应用 xff0c 现在搭建一个Springboot 43 Netty客户端的应用Demo程序 xff0c 多客户端方式
  • 机器学习中的凸和非凸优化问题

    题目 xff08 145 xff09 xff1a 机器学习中的优化问题 xff0c 哪些是凸优化问题 xff0c 哪些是非凸优化问题 xff1f 请各举一个例子 凸优化定义 凸优化问题 非凸优化问题 凸优化定义 xff1a 公式 geome
  • VMware workstation中rhel安装VMware tools失败

    切换登录用户为root即可 转载于 https www cnblogs com dazzleC p 10555809 html
  • Uniform convergence may be unable to explain generalization in deep learning

    本文价值 xff1a understand the limitations of u c based bounds cast doubt on the power of u c bounds to fully explain general
  • 调参之learning rate

    The learning rate is perhaps the most important hyperparameter If you have time to tune only one hyperparameter tune the
  • 调超参(lr,regularization parameter)经验整理

    Learning rate 最优值从1e 4到1e 1的数量级都碰到过 xff0c 原则大概是越简单的模型的learning rate可以越大一些 https blog csdn net weixin 44070747 article de
  • Dropout network, DropConnect network

    Notations input v v v output r r r weight parameter
  • Curriculum adversarial training

    Weakness of adversarial training overfit to the attack in use and hence does not generalize to test data Curriculum adve
  • Python处理中文语言——读取中文

    本文解决问题 xff1a 1 导入中文txt文本 xff0c 并转换为unicode 2 导入包含中文的py file 解决问题一 xff1a 导入中文txt文本 xff0c 并转换为unicode 基础概念 xff1a 1 unicode
  • C# WPF开源控件库HandyControl用法举例

    目录 概述 MessageBox用法举例 Button用法举例 Lable用法举例 Slider用法举例 TextBox用法举例 组合框ComboBox用法举例 源码下载 概述 HandyControl是一款免费开源的WPF控件库 xff0
  • python 等差数列生成器

    典型的迭代器模式作用很简单 遍历数据结构 不过 xff0c 即便不是从集合中获取元素 xff0c 而 是获取序列中即时生成的下一个值时 xff0c 也用得到这种基于方法的标准接口 例如 xff0c 内置的 range 函数用于生成有穷整数等
  • python 终止协程和异常处理

    协程中未处理的异常会向上冒泡 xff0c 传给 next 函数或 send 方法的调用方 xff08 即触发协程的对 象 xff09 下面示例举例说明如何使用之前博客示例中由装饰器定义的 averager 协程 未处理的异常会导致协程终止
  • centos7 下安装 nodejs

    源码包安装 下载安装包到 xff1a usr local 目录下 1 命令下载 wget https span class token punctuation span span class token operator span node
  • Ubuntu配置apt软件源

    清华大学开源镜像网站 xff08 帮助页面 xff09 https mirrors tuna tsinghua edu cn help AOSP 阿里云开源镜像网站 https opsx alibaba com mirror 网易开源镜像网
  • python3 fnmatch和fnmatchcase

    你想使用 Unix Shell 中常用的通配符 比如 py Dat 0 9 csv 等 去匹配文本字符串 xff0c fnmatch 模块提供了两个函数 fnmatch 和 fnmatchcase xff0c 可以用来实现这样的匹配 用法如
  • python unicodedata 处理Unicode 字符串

    你正在处理 Unicode 字符串 xff0c 需要确保所有字符串在底层有相同的表示 span class token comment coding utf 8 span span class token comment 你正在处理 Uni
  • python 插入排序

    问题 xff1a 数组排序 插入排序 xff0c 向已经有序一组序列中 xff0c 插入一个新的元素 默认第一个列表元素为已经排序好的元素 xff0c 从第二个元素进行比较 xff0c 已经排序好的元素 xff0c 重大到小 xff0c 依
  • 分治策略-归并排序

    问题 xff1a 数组排序 分治策略 归并排序 xff1a 1 是合并这些子问题的解 2 分解原问题 xff0c 递归求解 span class token comment coding utf 8 span span class toke
  • 求股票最大收益问题

    问题 xff1a 求股票最大收益 xff0c 股票每天的价格 xff1a 100 113 110 85 105 102 86 63 81 101 94 106 101 79 94 90 97 买进和卖出都在当天结束后进行 xff0c 在某一

随机推荐

  • Python pip 包的安装和卸载 使用。

    Python pip 包的安装和卸载 使用 xff08 一 xff09 pip 安装 一般 来说 Python 需要什么包 直接 pip install 包 即可 但是 这种方法太慢 因为他通过美国的服务器下载 提高 pip 速度 这里提供
  • jdk1.8安装和环境变量配置

    一 安装JDK 选择安装目录 安装过程中会出现两次 安装提示 第一次是安装 jdk xff0c 第二次是安装 jre 建议两个都安装在同一个java文件夹中的不同文件夹中 xff08 不能都安装在java文件夹的根目录下 xff0c jdk
  • python 读取PDF(tabula和pdfminer和pdfplumber的简单操作)

    一 pdfminer 读取PDF 官方文档 xff1a http www unixuser org euske python pdfminer 这里针对python3 1 模块安装 xff1a pip install i https pyp
  • 一区即将要洗的DVD片子

    101 Dalmatians Animated 2009 SE 101斑点狗 预计2009年发行特别版 12 Monkeys 05 10 2005 COM DOC 12只猴子 预计2005年5月10日发行扩展版 加评论和记录片等 2001
  • UML — 五大关系

    在UML教学视频中 xff0c 关系有四种 xff0c 而课本中有五种 xff0c 其实就是多加了一种 xff0c 那么下面我一并总结出来 1 关联关系 通俗点说就是关联关系就是两个对象他们之间的联系和关系 关联分两种 xff1a xff0
  • rhel6.5救援模式修复系统

    如果系统中很多重要的部分被删除了例如 boot下的所有东西 xff0c 则可以通过救援模式 root 64 dazzle1 桌面 mkdir backup root 64 dazzle1 桌面 cp etc fstab backup fst
  • 利用nvm安装npm失败的解决办法

    最近发现在安装nodejs后 xff0c 想使用npm发现自己的电脑上没有安装npm xff0c 可是网上都说安装了nodejs后会自动安装npm xff0c 找了很久解决办法发现没有合适的解决办法 xff0c 于是自己尝试了很久发现了问题
  • chrome 浏览器的缩略图怎么没有了?就是浏览过网页的缩略图,一点击就能打开网站。

    这个问题 xff0c 突然今天解决了 哈哈 分享 首先新标签页 点击左下角 最常访问的网站 点击 最常访问的网站 紧接着再点击被置顶端的 最常访问的网站 Ok xff0c 大功告成了 烦恼了几天的这个小功能 xff0c 有缩略图还是看着舒服
  • 史上最详细的PID教程——理解PID原理及优化算法

    Matlab动态PID仿真及PID知识梳理 云社区 华为云 huaweicloud com 位置式PID与增量式PID区别浅析 Z小旋 CSDN博客 增量式pid https zhuanlan zhihu com p 38337248 期望
  • ubuntu 20.04搭建samba文件共享服务器,实现基于Linux和Windows的共享文件服务

    ubuntu 20 04搭建samba文件共享服务器 xff0c 实现基于Linux和Windows的共享文件服务 超详细 一 xff0c samba的基本概念二 xff0c samba的安装三 xff0c samba的基本配置创建文件夹更
  • ERROR: Could not find a version that satisfies the requirement torchvision

    打docker时出错 xff0c ERROR Could not find a version that satisfies the requirement torchvision from versions 0 1 6 0 1 7 0 1
  • openstack 常用命令回顾及总结

    1 概述 命令实际执行基于OpenStack Queens版本 xff0c 更高版本亦可 xff0c 长时间未使用openstack有些遗忘 xff0c 整理后方便自己回顾学习 xff0c 仅供各位参考 xff0c 详细命令及参数可以参考o
  • TPMS方案 传感器 infineon篇 (SP35 SP37)

    TPMS方案 xff08 SP35 SP37 xff09 传感器 infineon篇 关于sp37无压力芯片目前已有方案 关于sp35传感器已经稳定出货 xff0c 欢迎咨询 硬件原理图 软件说明 xff1a 协议 调制方式 FSK 频率
  • sudo rosdep init 出现 ERROR: cannot download default sources list from:

    sudo rosdep init 出现 ERROR cannot download default sources list from 针对目前安装ROS出现一下指令的错误 span class token function sudo sp
  • 新装linux主机可以ping通,但是SSH无法登陆

    0 xff0c 新装一台linux主机 xff0c 可是ssh连接不上 xff0c 能ping通 怎么办呢 xff1f 1 xff0c 先查看一下防火墙状态 sudo ufw status 2 xff0c 关闭防火墙 sudo ufw di
  • tcp头以及ip头

    转自http www cnblogs com zzq919101 p 7866550 html 在网上找了很多有关tcp ip头部解析的资料 xff0c 都是类似于下面的结构 抽象出图文是这种结构 xff0c 但是在底层中数据到底是怎么传输
  • C++初阶 —— 入门/概念

    目录 一 xff0c 关键字 xff08 C 43 43 98 xff09 二 xff0c 命名空间 命名空间定义 命名空间使用 三 xff0c C 43 43 输入 输出 四 xff0c 缺省参数 五 xff0c 函数重载 六 xff0c
  • C++初阶 —— list类

    目录 一 xff0c list介绍 二 xff0c list的使用 构造函数 list iterator的使用 list capacity list element access list modifiers list迭代器失效 三 xff
  • C++初阶 —— stack/queue

    目录 一 xff0c 容器适配器 deque双端队列 二 xff0c stack栈 stack接口 stack模拟实现 三 xff0c queue队列 queue接口 queue模拟实现 四 xff0c priority queue优先级队
  • C++初阶 —— 模板进阶

    目录 一 xff0c 非类型模板参数 模板参数分类 二 xff0c 模板特化 函数模板特化 类模板特化 三 xff0c 模板分离编译 分离编译 链接失败原因 解决方法 附 模板优点 模板缺点 一 xff0c 非类型模板参数 模板参数分类 类