lesson05-C++模板

简介: lesson05-C++模板

泛型编程

我们先看一个代码:

看着是不是有点麻烦,我们有没有一种通用的办法,让编译器能够根据不同的类型自动生成不同的函数呢?有,就是模板。

泛型编程:

编写与类型无关的通用代码,是代码复用的一种手段。

而模板是泛型编程的基础

函数模板

我们先看代码:

template<typename T>
void Swap(T& a, T& b)
{
  T tmep = a;
  a = b;
  b = temp;
}

函数模板概念 :

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

我们以上面代码为例讲解:

首先,函数模板的格式:

template<typename T1,typename T2, ......>

返回值 函数名(参数列表){}

template 的意思就是模板。

typename是用来定义模板参数的关键字,也可以用class替代,但不可以使用struct。

T1,T2等就是编译器自己根据实参类型推导出来的,或者由我们显式实例化(后面说)。

接下来我们进行各种可能的实验。

错误原因是,编译器通过实参a将T推导为int类型,通过实参e将T推导为double类型,但是只有一个T,编译器无法确定他是int还是double,就会报错。

template<class T1, class T2>
void Swap(T1& a, T2& b)
{
  double temp = a;
  a = b;
  b = temp;
}

多了这样一个重载就没问题了,因为我们多了一个模板参数,则T1推导为int,T2推导为double。

换成Add函数还有另一种解决办法:强制类型转换

template<class T>
T Add(T& a, T& b)
{
  return a + b;
}

但是这里涉及引用的隐式类型转换,参照:C++入门,目录-引用,所以我们的Swap函数需要变点东西:

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

ret结果为4。

去掉const,就会出现权限放大的问题,在e强制类型转换时产生临时变量,具有常性,而函数的参数是变量,就会报错。

还有一种解决办法,叫做显式实例化

我们指定模板的模板参数,叫做显式实例化,编译器自己推导,叫做隐式实例化。

模板参数的匹配规则:

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

#include <iostream>
using namespace std;
int add(int a, int b)
{
  return a + b;
}
template<class T>
T add(T a, T b)
{
  return a + b;
}
int main()
{
  add(1, 2);
  add<int>(1, 2);
  return 0;
}

对于非模板函数和同名函数模板,如果调用函数时传参与非模板函数完全相同,那么就优先调用非模板函数,而不是调用函数模板实例化出来的函数;如果模板可以产生一个具有更好匹配的函数,那么将选择模板去实例化函数。

#include <iostream>
using namespace std;
int Add(int a, int b)
{
  return a + b;
}
template<class T1, class T2>
T1 Add(T1 a, T2 b)
{
  return a + b;
}
int main()
{
  cout << Add(1, 2) << endl;
  cout << Add(1.0, 2) << endl;
  return 0;
}

我们进入调试时,第二个Add函数是模板实例化出来的函数。

 

类模板

类模板定义格式:

template<typename T1,typename T2, ......>

class 类模板名字{}

我们可以使用类模板写一个动态顺序表,而且通过类模板,我们更加能够体会到他的魅力以及他的强大,如果我们使用C语言来写,而我们的要求是写出多种数据类型的顺序表,用C语言需要重复拷贝多次去修改类型,代码会有很多重复,但是有了模板就不一样了。

template<class T>
class Vector
{
  T* _Data;
  int _size;
  int _capacity;
public:
  Vector(int capacity = 4)
    :_size(0)
    ,_capacity(capacity)
    ,_Data(new T[_capacity])
  {}
  ~Vector()
  {
    if (_Data)
      delete[] _Data;
    _Data = nullptr;
    _size = _capacity = 0;
  }
  T& operator[](int pos)
  {
    if (pos >= 0)
    {
      assert(pos < _size && pos >= 0);
      return _Data[pos];
    }
  }
};

各种类型的对象我们都可以实例化出来。

类模板的实例化

类模板实例化与函数模板实例化不同,需要显式写出模板参数

也许你有这样的疑问,是不是因为我没有传参数,所以类模板无法推导T是什么,但是我们可以看看class 类名,有参数列表吗?显然没有,所以才需要我们显式去传模板参数。

同时我们需要注意的是:

类模板并不是真正的类,就像模板函数一样,只有将他实例化后才是函数,才是真正的类。

目录
相关文章
|
10月前
|
机器学习/深度学习 自然语言处理 算法
通义灵码开发者社区的构成——开发者群体
通义灵码开发者社区汇聚了来自不同背景的开发者,包括专业软件工程师、数据科学家、学生和业余开发者等,他们通过代码贡献、文档编写和社区活跃参与,共同推动社区的创新与发展,为通义灵码的广泛应用提供了坚实基础。
通义灵码开发者社区的构成——开发者群体
|
3月前
|
机器学习/深度学习 监控 算法
局域网行为监控软件 C# 多线程数据包捕获算法:基于 KMP 模式匹配的内容分析优化方案探索
本文探讨了一种结合KMP算法的多线程数据包捕获与分析方案,用于局域网行为监控。通过C#实现,该系统可高效检测敏感内容、管理URL访问、分析协议及审计日志。实验表明,相较于传统算法,KMP在处理大规模网络流量时效率显著提升。未来可在算法优化、多模式匹配及机器学习等领域进一步研究。
81 0
|
机器学习/深度学习 数据采集 算法
数据挖掘和机器学习算法
数据挖掘和机器学习算法
|
监控 安全
计算机硬件风扇和散热器维护
【7月更文挑战第28天】
154 2
|
存储 监控 Apache
Flink整库同步 如何把数据丢入到 datahub 中
Flink整库同步 如何把数据丢入到 datahub 中
|
消息中间件 Java Apache
使用Spring Boot实现与ActiveMQ的消息队列集成
使用Spring Boot实现与ActiveMQ的消息队列集成
|
自然语言处理 中间件 开发者
Django的国际化与本地化支持:打造多语言应用
【4月更文挑战第15天】Django,一款强大的Web框架,内置出色的支持国际化和本地化功能,使得创建多语言应用变得简单。本文介绍了国际化(i18n)与本地化(l10n)的概念,阐述了Django的相应机制,包括标记可翻译字符串、提取与翻译、设置语言和地区、本地化格式处理。遵循文中步骤,开发者能有效构建适应不同语言和地区需求的Web应用,提升用户体验。
|
JSON JavaScript 数据格式
如何控制权限,更加好的方法
如何控制权限,更加好的方法
|
机器学习/深度学习 数据采集 编解码
【MATLAB】史上最全的 18 种信号分解算法全家桶
【MATLAB】5 种高创新性的信号分解算法: https://mbd.pub/o/bread/ZJ6bkplp 【MATLAB】13 种通用的信号分解算法: https://mbd.pub/o/bread/mbd-ZJWZmptt 【MATLAB】史上最全的 18 种信号分解算法全家桶: https://mbd.pub/o/bread/ZJ6bkplq 其他类算法 【MATLAB】史上最全的11种数字信号滤波去噪算法全家桶: https://mbd.pub/o/bread/ZJiYlphx 【MATLAB】史上最全的9种频谱分析算法全家桶: https://mbd.pub/o/bread/Z
833 1
|
人工智能 iOS开发 MacOS
菜鸟打印组件系列-vue3快速接入
本文一开始介绍了菜鸟打印组件,相关名词,以及在打印方面的相关能力和使用时的注意点。接着举例基于vue3+pina+elementPlus的项目通过步骤1.使用pina 创建websoket相关处理的模块,处理打印机与浏览器长连接的过程。步骤2.按照菜鸟要求的模板语言,调整自己的打印模板,并且放在public文件夹下,步骤3.在列表页其中使用el-table组件,引入websoket模块,连接菜鸟组件以及批量打印,实现简单的批量打印功能
1162 0
菜鸟打印组件系列-vue3快速接入

热门文章

最新文章