【C++】模版(初阶)

目录

[一. 函数模版](#一. 函数模版)

[1. 格式 原理](#1. 格式 原理)

[2. 函数模版的实例化](#2. 函数模版的实例化)

[二. 类模板](#二. 类模板)


cpp 复制代码
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;
}

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

使用函数重载太麻烦

模具,填充不同类型,生成具体类型的代码

泛型编程:编写与类型无关的通用代码,是代码复用的一种手段。模板是泛型编程的基础

一. 函数模版

1. 格式 原理

template<typename T1, typename T2, ......, typename Tn>
返回类型 函数名(参数列表)
{ }

typename 是用来定义模板参数关键字也可以使用class,不能使用struct

模板参数作用范围:紧跟的 { }

cpp 复制代码
template<typename T> // 模板参数 -- 类型
void Swap(T& left, T& right)
{
	T temp = left;
	left = right;
	right = temp;
}

template<typename T1, typename T2>
T1 Func (const T1& x, const T2& y)
{
	cout << x << " " << y << endl;
	return x;
}

cpp 复制代码
int main()
{
	int a = 0, b = 1;
	double c = 1.1, d = 2.2;
	Swap(a, b);
	Swap(c, d);

	// Date d1(1949, 10, 1), d2(2015, 7, 3);

    Func(1, 2);
    Func(1, 2.2);
    
	return 0;
}

调用的是模版实例化出的函数(模板就是将本来应该我们做的重复的事情交给了编译器 )


C++库里面有。不需要写Swap,也不用写模版。直接使用

cpp 复制代码
int main()
{
	int a = 0, b = 1;
	double c = 1.1, d = 2.2;
	swap(a, b);
	swap(c, d);

	// Date d1(1949, 10, 1), d2(2015, 7, 3);

	return 0;
}

2. 函数模版的实例化

隐式实例化:让编译器根据实参 推演模板参数的实际类型
显示实例化:在函数名后的<>中 指定模板参数的实际类型

cpp 复制代码
template<typename T>
T Add(const T& left, const T& right)
{
	return left + right;
}

template<typename T>
T* Alloc(int n)
{
    return new T[n];
}

int main()
{ 
	int a1 = 10, a2 = 20;
	double d1 = 10.1, d2 = 20.2;
	cout << Add(a1, a2) << endl;
	cout << Add(d1, d2) << endl;

	// cout << Add(a1, d1) << endl; 报错:编译器无法确定T的类型

	// 解决方案1:用户自己强制转换
	cout << Add(a1, (int)d1) << endl;
	cout << Add((double)a1, d1) << endl;

	// 解决方案2:显示实例化
    cout << Add<int>(a1, d1) << endl;
	cout << Add<double>(a1, d1) << endl;

	// 有些函数无法自动推,只能显示实例化
	double* p1 = Alloc<double>(10);

	return 0;
}

类型转换会产生临时变量(常性)。Add(a1, (int)d1) 这里不是把 d1 传给 right,所以 Add 要加 const

二. 类模板

template<class T1, class T2, ..., class Tn>
class 类模板名
{
// 类内成员定义
};

普通类,类名和类型是一样
类模板,类名和类型不一样
类名:Stack
类型:Stack<T>

构造函数不一定用T这个参数,所以类模板都无法通过推演实例化,类都是显示实例化


cpp 复制代码
template<class T>
class Stack
{
public:
	Stack(int capacity = 4)
	{
		_a = new T[capacity];

		_capacity = capacity;
		_size = 0;
	}

	void Push(const T& Date)
	{
		// CheckCapacity();
		_a[_size] = Date;
		_size++;
	}

	~Stack()
	{
		if (_a)
		{
			free(_a);
			_a = nullptr;
			_capacity = 0;
			_size = 0;
		}
	}

private:
	T* _a;
	int _capacity;
	int _size;
};

int main()
{
	Stack<int> s1;    // int
	Stack<double> s2; // double
	Stack<char> s3;   // char
	
	// Stack<int, double>s4; 多个模板参数

	return 0;
}

类模板的声明和定义分离是别致的

类模板的声明和定义最好不要分离到 2个文件,会报错(后面讲)

cpp 复制代码
template<class T>
class Stack
{
public:
	Stack(int capacity = 4);

	void Push(const T& Date);

	~Stack()
	{
		if (_a)
		{
			free(_a);
			_a = nullptr;
			_capacity = 0;
			_size = 0;
		}
	}


private:
	T* _a;
	int _capacity;
	int _size;
};

template<class T>
Stack<T>::Stack(int capacity)
{
	_a = new T[capacity];

	_capacity = capacity;
	_size = 0;
}

template<class T>
void Stack<T>::Push(const T& data)
{
	// CheckCapacity();
	_a[_size] = data;
	_size++;
}

本篇的分享就到这里了,感谢观看 ,如果对你有帮助,别忘了点赞+收藏+关注

小编会以自己学习过程中遇到的问题为素材,持续为您推送文章

相关推荐
AKDreamer_HeXY15 分钟前
ABC434E 题解
c++·算法·图论·atcoder
罗湖老棍子15 分钟前
完全背包 vs 多重背包的优化逻辑
c++·算法·动态规划·背包
potato_may25 分钟前
C++ 发展简史与核心语法入门
开发语言·c++·算法
爱装代码的小瓶子1 小时前
【cpp知识铺子】map和set的前身-二叉搜索树
c++·算法
Embedded-Xin1 小时前
Linux架构优化——spdlog实现压缩及异步写日志
android·linux·服务器·c++·架构·嵌入式
[J] 一坚1 小时前
华为OD、微软、Google、神州数码、腾讯、中兴、网易有道C/C++字符串、数组、链表、树等笔试真题精粹
c语言·数据结构·c++·算法·链表
我不会插花弄玉1 小时前
c++入门基础【由浅入深-C++】
c++
不会编程的小寒2 小时前
C and C++
java·c语言·c++
hewayou2 小时前
MFC +Com+ALT工程报 内存泄漏
c++·mfc·内存泄漏·com技术
liulilittle2 小时前
C++ SSE/AVX/SHA/AES指令集检查,用于程序定向优化。
开发语言·c++·cpu·asm·detect·il·features