C++ ---- 模板

news2025/1/15 19:00:15

目录

泛型编程

函数模板

函数模板语法

模板使用

函数模板原理

函数模板的实例化

隐式实例化

显示实例化

模板参数的匹配原则

类模板

类模板的定义语法

类模板的实例化

非类型模板参数

类模板的特化

全特化

半特化(部分特化)

两个参数偏特化

模板的分离编译

模板总结


泛型编程

如下述代码,有需求要对int和double不同类型的数据进行交换。对同类型的问题重载了对应函数进行处理。

#include <iostream>
using namespace std;
void Swap(int& left, int& right)
{
	int temp = left;
    left = right;
	right = temp;
}

void Swap(double& left, double& right)
{
	double temp = left;
	left = right;
	right = temp;
}

int main()
{
	int a1 = 10, a2 = 20;
	double b1 = 13.14, b2 = 5.2;

	cout << "交换前:a1 = " << a1 << ", a2 = " << a2 << endl;
	cout << "交换前:b1 = " << b1 << ", b2 = " << b2 << endl;

	cout << "-------------------------------" << endl;

	Swap(a1,a2);
	Swap(b1,b2);

	cout << "交换后:a1 = " << a1 << ",a2 = " << a2 << endl;
	cout << "交换后:b1 = " << b1 << ",b2 = " << b2 << endl;
	return 0;
}

上述代码重载的函数仅仅是类型不同,代码的复用率较低。只要有新的类型出现,例如有新的需要要交换char类型的数据,就要增添对应的函数。

泛型编程的思想就是编写与类型无关的通用代码,是代码复用的一种手段。模板是泛型编程的基础。对于上述的场景可以使用模板来解决,根据不同的类型通过模板生成具体类型的代码。

函数模板

函数模板与类型无关,在使用时根据实参类型产生函数的特定类型版本,生成具体类型的代码。

函数模板语法

写法1:

template <class T1,class T2.....>
返回值类型 函数名 (){}

写法2:

template <typename T1,typename T2.....>
返回值类型 函数名(){}

上述用typename或者class来定义模板参数的写法都是可以的。

模板使用

对文章开头的问题用模板实现:

template<class T>
void Swap(T& left,T& right)
{
	T temp = left;
	left = right;
	right = temp;
}

int main()
{
	int a1 = 10, a2 = 20;
	double b1 = 13.14, b2 = 5.2;

	cout << "交换前:a1 = " << a1 << ", a2 = " << a2 << endl;
	cout << "交换前:b1 = " << b1 << ", b2 = " << b2 << endl;

	cout << "-------------------------------" << endl;

	Swap(a1,a2);
	Swap(b1,b2);

	cout << "交换后:a1 = " << a1 << ",a2 = " << a2 << endl;
	cout << "交换后:b1 = " << b1 << ",b2 = " << b2 << endl;
	return 0;
}

使用了模板以后,当要处理其他类型的数据时也不用增添新的代码,直接通过模板就可以生成一份处理新类型数据的代码。

函数模板原理

函数模板本身并不是函数,是编译器通过特定的方式产生具体类型函数的“模具”。所以模板就是将本来需要我们完成的任务交给了编译器。

在编译器编译阶段,编译器根据传入的实参类型推演生成对应类型的函数,供用户调用。

通过观察汇编代码可以发现,对于不同类型数据通过函数模板推演后,会生成不同的具体类型的函数代码。

函数模板的实例化

注意不要将函数模板的实例化和对象实例化联想在一起,对象实例化是根据类的大小开辟一块空间,再通过构造函数初始化。而函数模板的实例化是用不同类型的参数推演出具体类型的函数。

隐式实例化

让编译器根据实参推演模板参数的实际类型

template<class T>
T Add(const T& left, const T& right)
{
	return left + right;
}

int main()
{
	int a1 = 10, a2 = 20;
	double b1 = 1.1, b2 = 2.2;

	cout << Add(a1, a2) << endl;
	cout << Add(b1,b2) <<endl;
	
	return 0;
}

需要注意的是,在模板中,编译器一般不会进行类型转换操作,如下述示例,在推演阶段就会报错:


template<class T>
T add(const T& left,const T& right)
{
	return left + right;
}
int main()
{
	int a = 10;
	double b = 1.1;
	add(10,1.1);
	return 0;
}

 对于这个错误的解决,一般有两种处理方法,第一种就是在传参阶段用户做强制类型转换:

template<class T>
T add(const T& left,const T& right)
{
	return left + right;
}
int main()
{
	int a = 10;
	double b = 1.1;
	add(10,(int)1.1);
	add((double)10, 1.1);
	return 0;
}

第二种解决方法就是显示实例化。

显示实例化

在函数名后的<>指定模板参数的类型。

template<class T>
T Add(const T& left, const T& right)
{
	return left + right;
}

int main()
{
	int a = 10;
	double b = 13.14;

	Add<int>(a,b);
	return 0;
}

模板参数的匹配原则

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

template<class T>
T Add(T left,T right)
{
	cout << "T,模板函数" << endl;
	return left + right;
}

int Add(int left, int right)
{
	cout << "非模板函数" << endl;
	return left + right;
}

int main()
{
	Add(1,2);
	Add(1.1 , 2.2);
	return 0;
}

对于非模板函数和同名函数模板,如果在其他条件都相同。会优先调用非模板函数,因为编译器也很“懒”,已经有的非模板函数则不会在从模板生成一个。

template<class T>
T Add(T left,T right)
{
	cout << "T,模板函数" << endl;
	return left + right;
}

int Add(int left, int right)
{
	cout << "非模板函数" << endl;
	return left + right;
}

int main()
{
	Add(1,2);
	Add(3, 4);
	return 0;
}

总结:对于模板而言,如果已经有“现成”得非模板函数,就用现成的。如果需要模板产生一个更匹配的函数,会选择模板。

类模板

类模板的定义语法

template <class T>
class Stack
{
public:
    void PushBack(const T& data){}
private:
    T* _arr;
};

在上述代码中,Stack不是具体的类。是编译器根据被实例化的类型生成具体类的模具。

如果将成员函数定义在类外,也要加上模板参数列表,如:

template <class T>
class Stack
{
public:
	void Fun1(const T& x);
private:
	T* _arr;
};
template <class T>
void Stack<T>::Fun1(const T& x) {}

类模板的实例化

类模板的实例化与函数模板实例化不同,类模板实例化需要在类模板名后跟<>,显示的将要实例化的类型放在<>中,类模板名字不是真正的类,实例化的结果才是真正的类。如上述代码中,Stack不是真正的类,Statck<int>、Stack<char>、Stack<double>......才是真正的类。

非类型模板参数

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

template <class T,size_t N>
class Array
{
private:
	T _array[N];
	size_t _size;
};

需要注意的是,像浮点数、类对象、以及字符串是不允许作为非类型模板参数的。一般都是整数。非类型的模板参数必须在编译期间就能确认结果。

针对上述N的用法,在没介绍模板以前,用typedef好像也可以解决,如:

#define N 10
typedef int T;


class Array
{
private:
	T _array[N];
	size_t _size;
};

上述写法的弊端在于,当我们同时需要Array处理不同的数据,需要N不同大小的时候,就要手动的定义多份重复的代码。而模板会帮我们完成这样的工作,只需要我们给定类型和非类型模板参数就可以了。

类模板的特化

特化,就是对于一些特殊类型特殊处理。因为模板可以实现与类型无关的代码。可能对于部分类型的处理会产生错误的结果,基于这种场景就需要模板的特化。

全特化

class Date
{
public:
	Date(int year,int month,int day)
		:_year(year)
		,_month(month)
		,_day(day)
	{}
	bool operator>(const Date& d1)const
	{
		if (_year > d1._year)
		{
				return true;
		}
		else if (_year == d1._year)
		{
			if (_month > d1._month)
			{
				return true;
			}
		}
		else if (_year == d1._year && _month == d1._month)
		{
			if (_day > d1._day)
			{
				return true;
			}
		}

		return false;
	}
private:
	int _year;
	int _month;
	int _day;
};


template<class T>
struct A
{
	bool operator()(const T& x, const T& y)
	{
		return x > y;
	}
};

int main()
{
	A<int> a;
	cout << a(3, 2) << endl;
	Date d1(2001, 10, 3);
	Date d2(2023, 3, 2);

	A<Date> dd1;
	cout << dd1(d1, d2) << endl;

	A<Date*> dd2;
	cout << dd2(&d1, &d2) << endl;

	return 0;
}

针对Date*类型进行特化:

template<>
struct A<Date*>
{
	bool operator()(const Date* x, const Date* y)
	{
		return *x > *y;
	}
};

半特化(部分特化)

将模板参数类表中的一部分参数特化:

template <class T1,class T2>
class Data
{
public:
	Data() { cout << "<T1,T2>" << endl; }
private:
	T1 _d1;
	T2 _d2;
};

template <class T1>
class Data<T1, int>
{
public:
	Data() { cout << "Date<T1,int>" << endl; }
private:
	T1 _d1;
	int _d2;
};
int main()
{

	Data<double, int> d1;
	Data<char, int> d2;
	Data<double, char> d3;
	return 0;
}

通过测试结果我们可以发现,编译器也是很“懒”的,只要第二个参数匹配了int就走半特化的模板。

两个参数偏特化

template <class T1, class T2>
class Data
{
public:
	Data() { cout << "<T1,T2>" << endl; }
private:
	T1 _d1;
	T2 _d2;
};

template <class T1,class T2>
class Data<T1*, T2*>
{
public:
	Data() { cout << "<T1*,T2*>" << endl; }
private:
	T1 _d1;
	T2 _d2;
};


int main()
{
	Data<double, char> d1;
	Data<int*,int*> d2;
	return 0;
}

模板的分离编译

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

程序翻译图解

模板分离编译问题分析

代码:

main.cpp

#include "Add.h"

int main()
{
	Add(1,2);
	Add(1.1, 2.2);
	return 0;
}

Add.h

#pragma once
template<class T>
T Add(const T& left, const T& right);

Add.cpp

#define _CRT_SECURE_NO_WARNINGS 1
#include "Add.h"
template<class T>
T Add(const T& left, const T& right)
{
	return left + right;
}

链接报错:

分析:

在预处理阶段,会将Add.h在main.c中展开,main.c中有了Add函数的声明。汇编后链接前,main.o和Add.o都形成了各自的符号表。在最后链接过程中,main.o要根据Add函数名去Add.o中找到Add函数的地址。但是,问题出现了,在Add.cpp中,编译器没有看到对Add模板函数的实例化,因此没有生成具体的Add函数。所以上述代码会在链接时报错。

解决方法:

●将声明和定义放在一个文件下。

main.cpp

#include "Add.hpp"

int main()
{
	Add(1,2);
	Add(1.1, 2.2);
	return 0;
}

Add.hpp

#pragma once
template<class T>
T Add(const T& left, const T& right);

template<class T>
T Add(const T& left, const T& right)
{
	return left + right;
}

模板总结

模板优点:

●模板复用了代码,节省了资源,更快的迭代开发,在模板的基础上,C++的标准模板库STL因此而生。

●增强了代码的灵活性

模板缺点:

●可能带来代码膨胀的问题,编译时间变长。

●出现模板编译错误的时候,错误信息非常凌乱,不易定位错误。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/589659.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

排查Javascript内存泄漏案例(一)

Chrome DevTools里的Performance面板和Memory面板可以用来定位内存问题。 如何判断应用发生内存泄漏&#xff1f; 为了证明螃蟹的听觉在腿上&#xff0c;一个专家捉了只螃蟹并冲它大吼&#xff0c;螃蟹很快就跑了。然后捉回来再冲它吼&#xff0c;螃蟹又跑了。最后专家把螃蟹的…

浅谈 ChatGPT —— 现代巴别塔

theme: nico 一、用 ChatGPT 一搜就到你这了 ChatGPT 在去年 11 月发布以后&#xff0c;上线 5 天后就有了 100 万用户&#xff0c;上线两个月后已有上亿用户&#xff0c;可谓一炮而红。起初我对 ChatGPT 是没有什么感知的&#xff0c;我单纯认为人工智能还没有发展到完全超越人…

数字经济等相关概念与官方文档

一、数字经济 数字经济&#xff0c;作为一个内涵比较宽泛的概念&#xff0c;凡是直接或间接利用数据来引导资源发挥作用&#xff0c;推动生产力发展的经济形态都可以纳入其范畴。在技术层面&#xff0c;包括大数据、云计算、物联网、区块链、人工智能、5G通信等新兴技术。在应…

美食菜谱类APP小程序开发功能有哪些?

想要开发出一款靠谱好用的美食菜谱APP小程序系统&#xff0c;需要具备哪些基本功能呢&#xff1f; 1、视频教学。对于美食的教学教学方法最直接受用的就是视频教学&#xff0c;用户浏览起来更加方便而且可以直接跟着视频操作&#xff0c;效果更佳。用户也可以自己拍摄制…

JVM垃圾回收篇之相关概念和算法

垃圾回收相关概念 什么是垃圾 垃圾就是指在运行程序中没有任何指针指向的对象,这个对象就是需要被回收掉的垃圾,如果不及时进行清理,越积越多就会导致内存溢出. 为什么需要GC 不进行回收,早晚会导致内存溢出,Java自动管理垃圾回收,不需要开发人员手动干预,这就有可能导致开…

【记录】OLAP和OLTP

下面为提炼的重点内容&#xff0c;全部内容参考&#xff1a;OLAP和OLTP OLAP和OLTP的不同设计要求 对于OLTP系统和OLAP系统有哪些不同的要求&#xff1f;要说清楚这个&#xff0c;首先需要分析一下这两个系统的关键特征。 对于OLTP系统来说&#xff0c;最重要的事情是&#…

LIS系统字典模块功能

字典管理模块&#xff1a; 系统参数功能简介&#xff1a;集中设置系统使用过程中所需的参数值&#xff0c;一般由开发或实施人员进行设置。 标本管理功能简介&#xff1a;标本管理是对检验业务中涉及的检验标本类型进行初始化设置&#xff0c; …

Spring Cloud Zuul 是什么?如何使用它来实现 API 网关?

Spring Cloud Zuul 是什么&#xff1f;如何使用它来实现 API 网关&#xff1f; 在微服务架构中&#xff0c;服务之间的通信变得非常频繁和复杂。为了简化服务之间的通信和管理&#xff0c;很多企业都采用了 API 网关的架构模式。API 网关可以帮助我们统一处理服务的入口、路由…

【简单实用框架】【AddressablesMgr】【可移植】

☀️博客主页&#xff1a;CSDN博客主页&#x1f4a8;本文由 萌萌的小木屋 原创&#xff0c;首发于 CSDN&#x1f4a2;&#x1f525;学习专栏推荐&#xff1a;面试汇总❗️游戏框架专栏推荐&#xff1a;游戏实用框架专栏⛅️点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd;&#…

提升您的 MQTT 云服务:深入探索 BYOC

引言 您是否希望将物联网基础设施提升到更高的水平&#xff1f;为了应对业务的不断扩展&#xff0c;您需要一个强大且安全的消息平台来支持它。 MQTT 协议凭借其轻量级、发布/订阅模型和可靠性&#xff0c;已经成为构建物联网平台的首选方案。但是&#xff0c;随着业务的增长…

​力扣解法汇总1110. 删点成林

目录链接&#xff1a; 力扣编程题-解法汇总_分享记录-CSDN博客 GitHub同步刷题项目&#xff1a; GitHub - September26/java-algorithms: 算法题汇总&#xff0c;包含牛客&#xff0c;leetCode&#xff0c;lintCode等网站题目的解法和代码&#xff0c;以及完整的mode类&#…

Linux安装Redis数据库,实现远程连接

文章目录 1. Linux(centos8)安装redis数据库2. 配置redis数据库3. 内网穿透3.1 安装cpolar内网穿透3.2 创建隧道映射本地端口 4. 配置固定TCP端口地址4.1 保留一个固定tcp地址4.2 配置固定TCP地址4.3 使用固定的tcp地址连接 转发自cpolar内网穿透的文章&#xff1a;公网远程连接…

L-shape 方法

L-shape 方法是求解两阶段随机规划的一种常用方法&#xff0c;基本思想是利用切平面将第二阶段的反馈函数线性化&#xff0c;在构造切平面条件时有点类似 bender’s 方法。 注&#xff1a;这个图形中黑实线 Q ( x ) \mathcal{Q}(x) Q(x) 就是下面模型中的 L ( x ) \mathscr{L…

剑指 Offer 04. 二维数组中的查找解题思路

文章目录 标题解题思路优化 标题 在一个 n * m 的二维数组中&#xff0c;每一行都按照从左到右 非递减 的顺序排序&#xff0c;每一列都按照从上到下 非递减 的顺序排序。请完成一个高效的函数&#xff0c;输入这样的一个二维数组和一个整数&#xff0c;判断数组中是否含有该整…

ARM-伪操作

目录 协处理器指令 伪操作 安装交叉编译工具 Makefile 进入命令&#xff1a;vi ASM-ARM.s 宏定义 make之后查看ASM-ARM.dis反汇编文件 预编译指令 申请一个字的空间 .word 申请多个字节空间 嵌套编程 方式一&#xff1a;汇编跳转到C 方式二&#xff1a;C跳转到汇编 方式三&…

2023/5/30面试小结

一、本应会但并不会的题 1.用js写一个栈 class Stack {constructor() {this.items [];}// 添加元素到栈顶push(element) {this.items.push(element);}// 移除栈顶元素并返回pop() {return this.items.pop();}// 返回栈顶元素peak() {return this.items[this.items.length - …

HashMap有哪些线程安全的处理方式(面试题)

HashMap有哪些线程安全的处理方式&#xff08;面试题&#xff09; 概念1. synchronizedMap(Map<K,V> m)2. ConcurrentHashMapConcurrentHashMap Jdk1.7 vs Jdk1.8 概念 基于哈希表的 Map 接口的实现。此实现提供所有可选的映射操作&#xff0c;并允许使用 null 值和 null…

数字孪生3D可视化技术在数字化水利行业中的应用

城市供水数字孪生系统是一种基于web3d开发和数字孪生技术构建的智能运维系统&#xff0c;它可以将实际设备与虚拟模型相结合&#xff0c;实现对城市供水系统的实时监测、预测和优化。 智慧供水系统是智慧水务建设的新目标与新高度&#xff0c;能够实现城市内部原水供水以及污水…

ASEMI单向可控硅BT169D参数,BT169D规格,BT169D大小

编辑-Z 单向可控硅BT169D参数&#xff1a; 型号&#xff1a;BT169D 断态重复峰值电压VDRM&#xff1a;600V 平均通电电流IT(AV)&#xff1a;0.6A R.M.S通电电流IT(RMS)&#xff1a;0.8A 通态浪涌电流ITSM&#xff1a;10A 平均栅极功耗PG(AV)&#xff1a;0.1W 峰值门功率…

Curl【实例 01】curl下载使用及cmd实例脚本分享(通过请求下载文件)

Curl 官方下载地址 可下载不同平台不同版本的安装包。 这里写目录标题 1.是什么1.1 curl1.2 bat和cmd 2.常用命令2.1 发送GET请求并显示响应头信息2.2 显示详细的请求和响应信息2.3 限制重定向次数2.4 带Json数据的Post请求发送2.5 带文件的Post请求发送2.6 使用代理服务器发送…