返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++图文并茂分析讲解模板
  • 689
分享到

C++图文并茂分析讲解模板

2024-04-02 19:04:59 689人浏览 薄情痞子
摘要

目录1.内容引入2.模板函数C语言写交换函数c++写交换函数模板交换函数的语法及其原理语法原理理解显示实例化和隐式实例化关于编译器也是懒人这件事3.类模板1.内容引入 ​ 不知道大家

1.内容引入

​ 不知道大家是否在高中时背过英语范文模板,以下就是博主的回忆:

​ 这篇模板是一些英语比较好的老师写的。

​ 每当碰到感谢信时,我都会狂喜,尽管感谢的内容不同,地点不同,我都可以去根据模板,再根据作文分析模板的那些空对应应该填入什么。

其实呢c++中也用模板,但是这个时候,我们是写模板的人,而编译器变成了那个根据模板照葫芦画瓢的人

2.模板函数

C语言写交换函数

#include<iOStream>
using namespace std;
void Swapi(int* a, int* b)
{
	int tmp = *a;
	*a = *b;
	*b = tmp;
}
void Swapd(double* a, double* b)
{
	double tmp = *a;
	*a = *b;
	*b = tmp;
}
//……
int main()
{
	int a = 1, b = 2;
	Swapi(&a, &b);
	double c = 1.1, d = 2.2;
	Swapd(&c, &d);
	return 0;
}

​ 要实现不同类型的交换,实参不仅要传地址,而且不同类型的函数的名字要保持不同

至于为什么会这样,大家可以去看看我的文章。解释了为什么c语言不支持函数重载:

传送门

C++写交换函数

#include<iostream>
using namespace std;
void Swap(int& x, int& y)
{
	int tmp = x;
	x = y;
	y = tmp;
}
void Swap(double& x, double& y)
{
	double tmp = x;
	x = y;
	y = tmp;
}
//……
int main()
{
	int a = 1, b = 2;
	Swap(a, b);
	double c = 1.1, d = 2.2;
	Swap(c, d);
	return 0;
}

​ C++在语法上增加了引用和函数重载,在一定程度上弥补了c语言的不足,但是上述代码明明逻辑很相似,却还是要我们去实现不同类型的代码,对于我们这种懒人来说,简直就是煎熬

​ 但是计算机他是一个任劳任怨的好铁,不来不会感到疲劳,厌倦,是一个头脑优点笨笨的但是计算能力超强的大铁块。

模板交换函数的语法及其原理

语法

#include<iostream>
using namespace std;
template <class T>
void Swap(T& x, T& y)
{
	T tmp = x;
	x = y;
	y = tmp;
}
int main()
{
	int a = 1, b = 2;
	Swap(a, b);
	double c = 1.1, d = 2.2;
	Swap(c, d);
	return 0;
}

​ 这样写交换函数是不是就轻松多了,但是我们思考以下,上述代码调用的是一个Swap函数还是两个Swap函数呢?

回顾我们说的模板,是我们写的模板,然后编译器照着模板帮我们写出了intdouble类型的交换函数。

原理

图解:

我们也可以通过调试上述代码,转到反汇编,看看调用的函数是否真的是不同的函数。

理解显示实例化和隐式实例化

我们那模板加法函数来理解

#include<iostream>
using namespace std;
T Add(const T& x,const T& y)
{
	return x + y;
}
int main()
{
	int a = 1, b = 2;
	double c = 1.1, d = 2.2;
	cout << Add(a, b) << endl;//编译器要自己推类型的是隐式实例化
	cout << Add(c, d) << endl;
	//cout << Add(a, c) << endl;//error这样的写法就错了,为难编译器了,编译器也推不出来了
	cout << Add<int>(a, c) << endl;//不需要编译器去推的是显示实例化
	cout << Add<double>(b, d) << endl;
	cout << Add(a, (int)c) << endl;
	return 0;
}

编译器要自己去推T是什么类型的,就是隐式实例化

而由我们告诉编译器T是什么类型的,就是显示实例化

关于编译器也是懒人这件事

我们来看几道模板函数的代码来看看编译器是如何做事的:

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

​ 如果调试了上述代码就会发现,编译器第一次调用的是第一个Add函数,第二次由于我们的指定,编译器调用的是模板加法函数。

#include<iostream>
using namespace std;
int Add(int left, int right)
{
	return left + right;
}
template < class T1, class T2>
T1 Add(const T1 x,const T2 y)
{
	return x + y;
}
int main()
{
	Add(1, 2);
	Add(1, 2.0);//如果不写模板,会进行一个类型转换,再去调用第一个
	return 0;
}

3.类模板

由于c++的顺序表是用vector表示的,下面咱们的类名也用vector表示

像以前我们实现一个顺序表是这样的。

typedef int VDateType;
class vector
{
public:
	//……
private:
	VDateType* _a;
	size_t _size;
	size_t _capacity;
};
int main()
{
	vector v1;
	vector v2;
	return 0;
}

但是我们无法让v1是int类型的顺序表,v2是double类型的顺序表。

用模板类来实现

#include<iostream>
#include<assert.h>
using namespace std;
namespace kcc
{
	template<class T>
	class vector
	{
	public:
		vector()
			:_a(nullptr)
			, _size(0)
			, _capacity(0)
		{}
		// 拷贝构造和operator= 这里涉及深浅拷贝问题,还挺复杂,后面具体再讲
		~vector()
		{
			delete[] _a;
			_a = nullptr;
			_size = _capacity = 0;
		}
		void push_back(const T& x)
		{
			if (_size == _capacity)
			{
				int newcapacity = _capacity == 0 ? 4 : _capacity * 2;
				T* tmp = new T[newcapacity];
				if (_a)
				{
					memcpy(tmp, _a, sizeof(T) * _size);
					delete[] _a;
				}
				_a = tmp;
				_capacity = newcapacity;
			}
			_a[_size] = x;
			++_size;
		}
		// 读+写
		T& operator[](size_t pos);
		size_t size();
	private:
		T* _a;
		size_t _size;
		size_t _capacity;
	};
	// 模板不支持分离编译,也就是声明在.h ,定义在.cpp,原因后面再讲
	// 建议就是定义在一个文件 xxx.h  xxx.hpp
	// 在类外面定义
	template<class T>
	T& vector<T>::operator[](size_t pos)
	{
		assert(pos < _size);
		return _a[pos];
	}
	template<class T>
	size_t vector<T>::size()
	{
		return _size;
	}
}
int main()
{
	kcc::vector<int> v1;		// int
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	// v1.operator[](3);
	//cout << v1[3] << endl;
	//cout << v1[5] << endl;
	for (size_t i = 0; i < v1.size(); ++i)
	{
		v1[i] *= 2;
	}
	cout << endl;
	for (size_t i = 0; i < v1.size(); ++i)
	{
		cout << v1[i] << " ";
	}
	cout << endl;
	kcc::vector<double> v2;   // double
	v2.push_back(1.1);
	v2.push_back(2.2);
	v2.push_back(3.3);
	v2.push_back(4.4);
	for (size_t i = 0; i < v2.size(); ++i)
	{
		cout << v2[i] << " ";
	}
	cout << endl;
	return 0;
}

如果内部成员函数在类的外面定义的话,要加上类名::

当然了,本文章并不是重点介绍顺序表vector的实现,而是让大家看看类模板的效果

vector会在后续的文章中更新,敬请期待!

到此这篇关于C++图文并茂分析讲解模板的文章就介绍到这了,更多相关C++模板内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: C++图文并茂分析讲解模板

本文链接: https://lsjlt.com/news/167652.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • C++图文并茂分析讲解模板
    目录1.内容引入2.模板函数C语言写交换函数C++写交换函数模板交换函数的语法及其原理语法原理理解显示实例化和隐式实例化关于编译器也是懒人这件事3.类模板1.内容引入 ​ 不知道大家...
    99+
    2024-04-02
  • C++图文并茂分析讲解内存管理
    目录1.了解一些基本的内存段(图演示)验证栈是向下生长的验证堆一般是向上生长的(不一定)巩固内存管理知识点答案2.c++申请动态内存的新玩儿法new,delete回顾c语言动态内存管...
    99+
    2024-04-02
  • C++图文并茂讲解继承
    目录一、生活中的例子二、惊艳的继承三、继承的意义四、小结一、生活中的例子 组合关系∶整体与部分的关系 下面看一个组合关系的描述代码: #include <iostream&g...
    99+
    2024-04-02
  • Mybatis图文并茂讲解分页插件
    目录1. Mybatis分页插件1.1 分页插件介绍1.2 分页插件的使用1.3 分页插件的参数获取1.4 分页插件知识小结1. Mybatis分页插件 1.1 分页插件介绍 分页...
    99+
    2024-04-02
  • C语言图文并茂讲解分支语句用法
    目录一、if 语句分析二、switch 语句分析三、小结一、if 语句分析 if 语句用于根据条件选择执行语句else 不能独立存在且总是与它最近的 if 相匹配else 语句后可以...
    99+
    2024-04-02
  • C++图文并茂讲解类型转换函数
    目录一、类型转换函数(上)1.再论类型转换2.问题3.再论构造函数4.另一个视角5.编译器的行为6.小结(上) 二、类型转换函数(下)1.类型转换2.编译器的行为3.注意事...
    99+
    2024-04-02
  • 图文并茂讲解RocketMQ消息类别
    目录1、同步消息2、异步消息3、单向消息1、同步消息 即时性较强,重要的消息,且必须有回执的消息,例如短信,通知(转账成功) 生产者: public class Producer ...
    99+
    2022-12-27
    RocketMQ消息类别 RocketMQ消息类型
  • 图文并茂地讲解Mysql索引(index)
    目录前言1. 索引概述1.1 什么是索引?1.2 使用索引和不使用索引的区别1.3 索引的特点2. 索引结构2.1 概述2.2 二叉树2.3 B-Tree2.4 B+Tree2.5 ...
    99+
    2022-11-13
    mysql怎么使用索引 mysql的索引 mysql索引代码
  • SpringBoot图文并茂讲解登录拦截器
    目录1.相关概念1.实现效果2.实现步骤2.代码实现1.配置文件2.java代码3.前端代码3.运行测试1.相关概念 1.实现效果 当没有输入正确的账号密码登录成功时, 除了登录页,...
    99+
    2024-04-02
  • SpringBoot图文并茂讲解依赖管理的特性
    目录1.父依赖parent介绍2.修改默认版本号3.starter场景启动器1.父依赖parent介绍 pom文件中含有父依赖 <parent> ...
    99+
    2024-04-02
  • MyBatis图文并茂讲解注解开发多对多查询
    目录MyBatis的注解实现复杂映射开发多对多查询多对多查询的模型多对多查询的语句添加CourseMapper 接口方法使用注解配置Mapper测试类多对多配置总结MyBatis的注...
    99+
    2024-04-02
  • MyBatis图文并茂讲解注解开发一对多查询
    目录MyBatis的注解实现复杂映射开发一对多查询一对多查询的模型一对多查询的语句创建StudentMapper接口使用注解配置Mapper测试类一对多配置总结MyBatis的注解实...
    99+
    2024-04-02
  • MyBatis图文并茂讲解注解开发一对一查询
    目录MyBatis的注解实现复杂映射开发一对一查询一对一查询的模型一对一查询的语句创建PersonMapper接口使用注解配置Mapper测试类一对一配置总结MyBatis的注解实现...
    99+
    2024-04-02
  • SpringBoot图文并茂讲解Lombok库的安装与使用
    目录1.相关介绍2.安装步骤1.添加依赖2.安装插件3.使用注解1.相关介绍 Lombok是一个通过注解以达到减少代码的Java库,如通过注解的方式减少get,set方法,构造方法等...
    99+
    2024-04-02
  • C语言图文并茂详解链接过程
    目录一、链接器的意义二、模块链接三、小结一、链接器的意义 连接器的主要作用是把各个模块之间相互引用的部分处理好,使得各个模块之间能够正确的衔接。 二、模块链接 静态链接 由链接器在...
    99+
    2024-04-02
  • php案例:$_SERVER详解(图文并茂)
    作者:陈业贵 华为云享专家 51cto(专家博主 明日之星 TOP红人) 阿里云专家博主 文章目录 前言$_SERVER['PHP_SELF']$_SERVER['SERVER_ADDR...
    99+
    2023-09-08
    php 服务器 apache
  • C语言图文并茂详解程序编译过程
    目录一、初识编译器二、程序被编译的过程三、小结一、初识编译器 编译器是一个广义的概念,真正的编译器由下面几个模块组成,真正的编译器是进行语法分析和语义分析的。 二、程序被编译的过程...
    99+
    2024-04-02
  • Java图文并茂讲解主方法中的String[]args参数作用
    目录一、作用二、在控制台传入参数三、在IntelliJ IDEA中传入参数总结一、作用 主方法就是程序的入口,那么里面的String[] args参数是什么意思呢? String[]...
    99+
    2024-04-02
  • C++图文并茂轻松进阶面向对象
    目录一、进阶面向对象(上)二、小结(上)三、进阶面向对象(下)四、小结(下)一、进阶面向对象(上) 面向对象的意义在于 将日常生活中习惯的思维方式引入程序设计中将需求中的概念直观的映...
    99+
    2024-04-02
  • C++函数模板和类模板分析
    本篇内容介绍了“C++函数模板和类模板分析”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1.函数模板函数模板定义了参数化的非成员函数,这使得...
    99+
    2023-06-17
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作