【C++】---模板初阶(超详练气篇)

这篇具有很好参考价值的文章主要介绍了【C++】---模板初阶(超详练气篇)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

个人主页:平行线也会相交💪
欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 平行线也会相交 原创
收录于专栏【C++之路】💌
本专栏旨在记录C++的学习路线,望对大家有所帮助🙇‍
希望我们一起努力、成长,共同进步。🍓
【C++】---模板初阶(超详练气篇)

一、 泛型编程

泛型编程,好家伙,名字倒是挺吓人,其实并没有啥可怕的,我们通过变换函数来引入泛型编程,请看:

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;
}

上述代码中,我们可以对int类型、double类型、char类型进行交换,在未来,我们如果对更多类型的变量进行交换应该怎么呢?
这个是问题刚好可以利用泛型参数来解决,请看;

//模板
//函数模板+类
template<typename T>
void Swap(T& x1, T& x2)
{
	T tmp = x1;
	x1 = x2;
	x2 = tmp;
}
int main()
{
	int a = 10, b = 20;
	double c = 1.1, d = 2.2;
	Swap(a, b);
	Swap(c, d);
	return 0;
}

模板参数定义的是函数的参数类型。
另外一点,Swap(a, b);和Swap(c, d);调用的并不是同1个函数,我们观察一下汇编代码来看一下:
【C++】---模板初阶(超详练气篇)

如果我们想要交换两个指针,也是可以的,请看:int* p1 = &a;int* p2 = &b;swap,如下图我们可以看到也可以交换指针:
【C++】---模板初阶(超详练气篇)

二、 函数模板

模板概念:

函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,根据实参类型产生函数的特定类型版本。

2.1函数模板示例化

函数模板格式:

//返回值类型 函数名(参数列表){}
template<typename T>
void Swap( T& left, T& right)
{
 T temp = left;
 left = right;
 right = temp;
}

上述代码是交换函数的一个代码,那我们调用的时候并不是调用的那个模板,编译器调用的其实是函数模板实例化生成的具体函数。
编译器通过模板生成具体函数的过程称之为函数模板的实例化。请看图:
【C++】---模板初阶(超详练气篇)

在编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。比如:当用double类型使用函数模板时,编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码,对于字符类型也是如此。

template<typename T>
void Swap(T& x1, T& x2)
{
	T tmp = x1;
	x1 = x2;
	x2 = tmp;
}

下面举多个参数类型的例子,请看:

template<typename T1,typename T2>
T1 Func(const T1& x, const T2& y)
{
	cout << x << " " << y << endl;
	return x;
}
//T1  T2可以是任意类型,比如自定义类型或者内置类型
//函数模板实例化生成具体函数
//函数模板根据调用会自己推导模板参数的类型,实例化出对应的函数
int main()
{
	Func(1, 2);
	Func(1.1, 2.2);
	return 0;
}

【C++】---模板初阶(超详练气篇)
注意:template<typename T1>中的typename可以换成class,但是不可以换成struct

2.2函数模板实例化的两种方式

函数模板的实例化的两种方式,分为:隐式实例化和显式实例化。
【C++】---模板初阶(超详练气篇)
请注意看T Add(const T& left, const T& right)中为什么要加const,这里涉及到权限的问题。举个例子:cout << Add(a1, (int)d1) << endl;(隐式类型转换) cout << Add<double>(a1, d1) << endl; (显式类型转换)无论是显式类型的转换还是显式类型的转换,类型的转换会产生临时变量,临时变量具有常属性,常属性不能传给普通的引用(因为权限被放大了),所以要加上const

我能不能这样写呢?请看:
【C++】---模板初阶(超详练气篇)
现在我们应该怎么解决呢?首先这里解决方法有两种。
方式一(两者之间只能听其中一个的话),实参传递的时候推演T的类型
【C++】---模板初阶(超详练气篇)
方式二(显式实例化)
【C++】---模板初阶(超详练气篇)
上述的显式示例化其实并不是这样完的,下面来看一下必须要用到显式示例化的场景。
【C++】---模板初阶(超详练气篇)

三、类模板

3.1类模板的实例化

类模板的定义格式:

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

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

template<class T>
class Stack
{
public:
	Stack(size_t capacity = 10)
	{
		/*_array = (T*)malloc(capacity * sizeof(T));
		if (nullptr == _array)
		{
			perror("malloc申请空间失败");
			return;
		}*/
		_array = new T[capacity];
		_size = 0;
		_capacity = capacity;
	}
	void Push(const T& data)
	{
		// CheckCapacity();
		_array[_size] = data;
		_size++;
	}
	~Stack()
	{
		if (_array)
		{
			free(_array);
			_array = nullptr;
			_capacity = 0;
			_size = 0;
		}
	}
private:
	T* _array;
	size_t _size;
	size_t _capacity;
};
int main()
{

	Stack<int> s1;//s1存int类型
	Stack<double> s2;//s2存double类型
	Stack<char> s3;//s3存char类型
	return 0;
}

3.2类模板的声明和定义分离

对于普通类而言,类名就是类型。
对于类模板而言,类名是类名,类型是类型,二者是不一样的。
请看下面举例:

template<class T>
class Stack
{
public:
	Stack(size_t capacity = 10);
	void Push(const T& data);
	~Stack();
private:
	T* _array;
	size_t _size;
	size_t _capacity;
};

template<class T>
Stack<T>::Stack(size_t capacity)
{
	/*_array = (T*)malloc(capacity * sizeof(T));
	if (nullptr == _array)
	{
		perror("malloc申请空间失败");
		return;
	}*/
	_array = new T[capacity];
	_size = 0;
	_capacity = capacity;
}

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

template<class T>
Stack<T>::~Stack()
{
	if (_array)
	{
		free(_array);
		_array = nullptr;
		_capacity = 0;
		_size = 0;
	}
}

对于上述代码而言,Stack是类名,Stack<T>是类型,但是如果类模板实例化时候,类名就是具体的Stack<int>或者Stack<char>等

关于类模板的声明和定义分离的讲解,在模板进阶那一部分进行更深一步的讲解。

好了,以上就是【C++】模板初阶内容的讲解,就到这里啦!
再见啦,友友们!!!
【C++】---模板初阶(超详练气篇)文章来源地址https://www.toymoban.com/news/detail-476238.html

到了这里,关于【C++】---模板初阶(超详练气篇)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 算法修炼之练气篇——练气五层

    博主:命运之光 专栏:算法修炼之练气篇 前言:每天练习五道题,炼气篇大概会练习200道题左右,题目有C语言网上的题,也有洛谷上面的题,题目简单适合新手入门。(代码都是命运之光自己写的,练完这200多道题就考了今年第十四届的B组蓝桥杯C/C++获得了省一,后面还会

    2024年02月05日
    浏览(24)
  • 算法修炼之练气篇——练气十九层

    博主:命运之光 专栏:算法修炼之练气篇 前言:每天练习五道题,炼气篇大概会练习200道题左右,题目有C语言网上的题,也有洛谷上面的题,题目简单适合新手入门。(代码都是命运之光自己写的,练完这200多道题就考了今年第十四届的B组蓝桥杯C/C++获得了省一,后面还会

    2024年02月05日
    浏览(64)
  • 算法修炼之练气篇——练气十五层

    博主:命运之光 专栏:算法修炼之练气篇 前言:每天练习五道题,炼气篇大概会练习200道题左右,题目有C语言网上的题,也有洛谷上面的题,题目简单适合新手入门。(代码都是命运之光自己写的,练完这200多道题就考了今年第十四届的B组蓝桥杯C/C++获得了省一,后面还会

    2024年02月04日
    浏览(26)
  • 算法修炼之练气篇——练气十二层

    博主:命运之光 专栏:算法修炼之练气篇 前言:每天练习五道题,炼气篇大概会练习200道题左右,题目有C语言网上的题,也有洛谷上面的题,题目简单适合新手入门。(代码都是命运之光自己写的,练完这200多道题就考了今年第十四届的B组蓝桥杯C/C++获得了省一,后面还会

    2024年02月05日
    浏览(64)
  • 算法修炼之练气篇——练气二十二层

    博主:命运之光 专栏:算法修炼之练气篇 前言:每天练习五道题,炼气篇大概会练习200道题左右,题目有C语言网上的题,也有洛谷上面的题,题目简单适合新手入门。(代码都是命运之光自己写的,练完这200多道题就考了今年第十四届的B组蓝桥杯C/C++获得了省一,后面还会

    2024年02月04日
    浏览(31)
  • 【C/C++】之内存管理(超详细练气篇)

    个人主页:平行线也会相交💪 欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 平行线也会相交 原创 收录于专栏【C++之路】💌 本专栏旨在记录C++的学习路线,望对大家有所帮助🙇‍ 希望我们一起努力、成长,共同进步。🍓 程序中需要存储一些数据 ,那关于数据的分类主要

    2024年02月07日
    浏览(23)
  • 【C++初阶】第六站 : 模板初阶

    前言: 本章知识点:泛型编程、函数模板、类模板 专栏: C++初阶 目录 泛型编程 函数模板 1.函数模板概念 2.函数模板格式 3.函数模板的原理 4.函数模板的实例化 5.模板参数的匹配原则 类模板 类模板的定义格式 类模板的实例化 如何实现一个通用的交换函数呢? 代码如下:

    2024年03月17日
    浏览(36)
  • 【C++】模板初阶——函数模板和类模板

    🚀 作者简介:一名在后端领域学习,并渴望能够学有所成的追梦人。 🚁 个人主页:不 良 🔥 系列专栏:🛸C++  🛹Linux 📕 学习格言:博观而约取,厚积而薄发 🌹 欢迎进来的小伙伴,如果小伙伴们在学习的过程中,发现有需要纠正的地方,烦请指正,希望能够与诸君一同

    2024年02月10日
    浏览(30)
  • 【C++】模板初阶 【 深入浅出理解 模板 】

    如何实现一个通用的交换函数呢? 使用函数重载虽然可以实现 ,但是有一下几个不好的地方: 重载的函数 仅仅是类型不同 ,代码复用率比较低, 只要有新类型出现时,就需要用户自己增加对应的函数 代码的可维护性比较低,一个出错可能所有的重载均出错 那能否 告诉编

    2024年02月05日
    浏览(36)
  • c++之模板初阶

    💗 💗 博客:小怡同学 💗 💗 个人简介:编程小萌新 💗 💗 如果博客对大家有用的话,请点赞关注再收藏 🌞 函数重载的缺点 重载的函数仅仅是类型不同,代码复用率比较低,只要有新类型出现时,就需要用户自己增加对应的函 数 代码的可维护性比较低,一个出错可能所有

    2024年02月08日
    浏览(42)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包