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

【C++模板】

模板初阶

  • 前言
  • 1.定义模板
  • 2.函数模板
    • 2.1定义
    • 2.2实例化函数模板
    • 2.3模板参数的匹配原则
  • 3.类模板
    • 3.1类模板实例化

前言

模板是C++中泛型编程的基础,一个模板就是一个创建类和函数的蓝图或公式。

1.定义模板

假定我们希望编写一个函数来比较两个值,并指出第一个值是小于,等于还是大于第二个值。我们可以写多个函数重载去比较不同类型的两个值。不仅麻烦繁琐,而且还要考虑所有类型。

2.函数模板

2.1定义

我们可以定义一个通用的函数模板,而不是为每个类型都定义一个新函数。一个模板就是一个公式,可用来生成针对特定类型的函数版本,例如上文的比较两个值。

//如果两个值相等,返回0,如果v1小返回-1,如果v2小返回1
template <typename T>
int compare(const T& v1, const T& v2)
{
	if (v1 < v2) return -1;
	if (v1 > v2) return 1;
	return 0;
}

模块定义以关键字template开始,后跟一个模块参数列表,用逗号分隔的一个或多个模块参数的列表,用<,>包围起来。

2.2实例化函数模板

用不同的类型,使用函数模板,就叫做模板参数实例化。模板参数实例化分为隐式实例化和显示实例化。
(1).隐式实例化:让编译器根据实参推演模板参数的实际类型

template<class T>
T Add(const T & a, const T& b)
{
	return a + b;
}
int main()
{
	int x = 0;
	int y = 1;
	double c = 1.2;
	double d = 2.1;
	Add(x, y);
	Add(c, d);
	Add(x, (int)d);
	//有问题,因为模板参数列表只有一个T,无法判断是int 还是 double 
	//所以此时选择强转或者显示类型转换
	return 0;

(2).显示实例化:在函数名后的<>中指定模板参数的实际类型

int main(void)
{
	int a = 10;
	double b = 20.0;
	// 显式实例化
	Add<int>(a, b);
	return 0;
}

如果类型不匹配,编译器会尝试进行隐式类型转换,如果无法转换成功编译器将会报错。

2.3模板参数的匹配原则

(1).一个非模板函数可以和一个同名的模板函数同时存在,而且该函数还可以被实例化成这个非模板函数。

int Add(int left, int right)
{
	return left + right;
}
// 通用加法函数
template<class T>
T Add(T left, T right)
{
	return left + right;
}
void Test()
{
	Add(1, 2); // 与非模板函数匹配,编译器不需要特化
	Add<int>(1, 2); // 调用编译器特化的Add版本
}

(2).对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模板产生出一个实例。如果模板可以产生一个具有更好匹配的函数, 那么将选择模板。
(3).模板函数不允许自动类型转换,但普通函数可以进行自动类型转换。

3.类模板

类模板是用来生成类的蓝图的,与函数模板不同的是,编译器不能为类模板推断类模板的参数类型。

template<class T1, class T2, ..., class Tn>
class 类模板名
{
	// 类内成员定义
};
#include<iostream>
using namespace std;
// 类模版
template<typename T>
class Stack
{
public:
	Stack(size_t capacity = 4)
	{
		_array = new T[capacity];
		_capacity = capacity;
		_size = 0;
	}
	void Push(const T& data);
private:
	T* _array;
	size_t _capacity;
	size_t _size;
};
template<class T>
void Stack<T>::Push(const T& data)

3.1类模板实例化

类模板实例化与函数模板实例化不同,类模板实例化需要在类模板名字后跟<>,然后将实例化的类型放在<>中即可,类模板名字不是真正的类,而实例化的结果才是真正的类。

Stack<int> st1; // int
Stack<double> st2; // double

http://www.kler.cn/a/593667.html

相关文章:

  • Linux--进程创建
  • Android 13深度定制:SystemUI状态栏时间居中显示终极实战指南
  • The Illustrated Stable Diffusion
  • 电机控制常见面试问题(十四)
  • pytorch v1.4.0安装问题
  • 2024年国赛高教杯数学建模E题交通流量管控解题全过程文档及程序
  • 嵌入式系统开发如何选择和备考软考高级
  • webpack等构建工具如何支持移除未使用的代码
  • 基于carla的模仿学习(附数据集CORL2017)更新中........
  • WPF 中的 GridSplitter 详解
  • 不使用负压电源,ADC如何测量正负压?
  • 为什么渲染农场渲染帧而非视频?核心原因 + 举例
  • Neo4j GDS-02-graph-data-science 简单聊一聊图数据科学插件库
  • 计算机网络基础:设计高效的网络布局
  • 使用cartographer扩展地图
  • 【Linux】VMware 17 安装 VMware Tools
  • 网络运维学习笔记(DeepSeek优化版) 019 HCIA-Datacom新增知识点01网络管理与运维
  • docker 创建mysql5.7 并开启bin_log和general_log日志审计功能
  • docker 内 pytorch cuda 不可用
  • 【JavaEE】传递和接收数据,Spring MVC 注解搭建前后端交互的「隐形桥梁」