C++好难(6):模板初阶

news/2024/10/24 0:17:31/

【本节目标】

  • 1. 泛型编程
  • 2. 函数模板
  • 3. 类模板

目录

【本节目标】

1.泛型编程

2.函数模板

概念:

格式:

原理:

实例化:

1.隐式实例化:

2.显式实例化

原则一:

原则二:

原则三:

3.类模板

格式

类模板的实例化


1.泛型编程

如何实现一个通用的交换函数呢?

以下面的交换函数为例:

// 交换两个整型变量
void Swap1(int& p1, int& p2) 
{int tmp = p1;p1 = p2;p2 = tmp;
}// 交换两个字符型变量
void Swap(char& p1, char& p2) 
{char tmp = p1;p1 = p2;p2 = tmp;
}

可以看到两种不同类型的交换函数的实现,我们用重载函数去实现的

使用函数重载虽然可以实现,但是有一下几个不好的地方:

  • 1. 重载的函数仅仅是类型不同,代码复用率比较低,只要有新类型出现时,就需要用户自己增加对应的函数
  • 2. 代码的可维护性比较低,一个出错可能所有的重载均出错

如果我们像印刷一样,弄一个模具,然后让编译器根据不同的类型利用该模具来生成代码呢?

如果在C++中,也能够存在这样一个模具,通过给这个模具中填充不同材料(类型),来获得不同材料的铸件(即生成具体类型的代码),那将会节省许多头发。巧的是前人早已将树栽好,我们只需在此乘凉。

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

2.函数模板

概念:

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

格式:

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

注意:typename 是用来定义模板参数关键字,也可以使用 class(切记:不能使用 struct 代替 class)

这里我们推荐使用class,因为它短,而且STL里面用的就是class

原理:

函数模板是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。

所以其实模板就是将本来应该我们做的重复的事情交给了编译器。

在编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。

比如:当用double类型使用函数模板时编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码,对于字符类型也是如此。

实例化:

用不同类型的参数使用函数模板时,称为函数模板的实例化。

模板参数实例化分为:隐式实例化和显式实例化。

1.隐式实例化:

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

template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;double d1 = 1.1, d2 = 2.2;Add(a1, a2); // 编译器根据实参a1和a2推演出模板参数为int类型Add(d1, d2); // 编译器根据实参d1和d2推演出模板参数为int类型return 0;
}

但是注意以下情况属于编译失败:

因为a1是int,d1是couble,编译器无法确定此处到底该将 T 确定为 int 或者 double 类型而报错。

处理这种方式有两种方法:

1)用户自己来进行强制转换

template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;double d1 = 1.1, d2 = 2.2;Add((double)a1, d1); // 把a1强转成doubleAdd(a1, (int)d1); // 把d1强转成intreturn 0;
}

2)使用个显示实例化

template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;double d1 = 1.1, d2 = 2.2;Add<int>(a1, d1); Add<double>(a1, d1);return 0;
}

总结:这个T必须是明确的

2.显式实例化

所谓显示实例化,就是在函数名后的  < >  中指定模板参数的实际类型。

template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;Add<int>(a1, a1); return 0;
}

原则一:

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

// 专门处理int的加法函数
int Add(int left, int right)
{cout << "非模板调用" << endl;return left + right;
}// 通用加法函数
template<class T>
T Add(T left, T right)
{cout << "模板调用" << endl;return left + right;
}int main()
{cout << Add(1, 2) << endl; // 与非函数模板类型完全匹配,不需要函数模板实例化cout << Add<int>(1, 2) << endl; // 模板函数可以生成更加匹配的版本,编译器根据实参生成更加匹配的Add函数return 0;
}

编译器会优先去选择已经存在的函数,如果没有这个函数的存在,才会去用函数模板生成

原则二:

2)对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模板产生出一个实例。

如果模板可以产生一个具有更好匹配的函数, 那么将选择模板

// 专门处理int的加法函数
int Add(int left, int right)
{cout << "非模板调用" << endl;return left + right;
}// 通用加法函数
template<class T>
T Add(T left, T right)
{cout << "模板调用" << endl;return left + right;
}int main()
{cout << Add(1, 2) << endl; // 与非函数模板类型完全匹配,不需要函数模板实例化cout << Add(1.1, 2.0) << endl; // 模板函数可以生成更加匹配的版本,编译器根据实参生成更加匹配的Add函return 0;
}

原则三:

模板函数不允许自动类型转换,但普通函数可以进行自动类型转换。

3.类模板

格式

类模板格式:

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

以一个栈的数据结构定义为例:

template<class T>
class Stack
{
public:Stack(int capaicty = 4):_a(new T[capaicty]),_top(0), _capacity(capaicty){}~Stack(){delete[] _a;_capacity = _top = 0;}private:T* _a;size_t _top;size_t _capacity;
};

类模板的实例化

类模板实例化与函数模板实例化不同,类模板实例化需要在类模板名字后跟 < >然后将实例化的类型放在 < > 中即可。

// 跟上述栈定义的 代码
int main()
{Stack<int> st1; // intStack<double> st2; // doublereturn 0;
}

类模板名字不是真正的类,而实例化的结果才是真正的类


http://www.ppmy.cn/news/63769.html

相关文章

CentOS 7(2009) 升级 GCC 版本

1. 前言 CentOS 7 默认安装的 gcc 版本为 4.8&#xff0c;但是很多时候都会需要用到更高版本的 gcc 来编译源码&#xff0c;那么本文将会介绍如何在线升级 CentOS 的 gcc 版本。 2. 升级 GCC (1). 安装 centos-release-scl&#xff1b; [imaginemiraclecentos7 ~]$ sudo yum…

C语言:指针求解鸡兔同笼问题

题目&#xff1a;鸡兔同笼问题 要求&#xff1a;使用自定义函数void calc(int h, int f,int *c,int *r) 求解鸡兔同笼问题。 h 表示总的头数&#xff0c;f 表示总的脚数。 例子&#xff1a; 输入&#xff1a; 5 16 输出&#xff1a; 2 3 分析&#xff1a; 在该代码中&a…

即时通讯APP开发费用成本多少?

移动互联网的发展&#xff0c;为人们的通讯交流提供了非常多的便利&#xff0c;一些即时通讯APP的出现&#xff0c;将人与人的距离再一次缩短。通过即时通讯APP软件&#xff0c;人们可以随时随地了解身边发生的新鲜事物&#xff0c;以及和朋友探讨各类趣事&#xff0c;甚至可以…

HTTP第三讲——四层模型、七层模型

四层模型 TCP/IP 协议&#xff0c;它是 HTTP 协议的下层协议&#xff0c;负责具体的数据传输 工作。TCP/IP 协议是一个“有层次的协议栈”。 TCP/IP 当初的设计者真的是非常聪明&#xff0c;创造性地提出了“分层”的概念&#xff0c;把复杂的网络通信划分出多个层次&#xff…

数据结构——结构体 内存对齐

在C语言中&#xff0c;可以使用结构体&#xff08;Struct&#xff09;来存放一组不同类型的数据。结构体是一种集合&#xff0c;它里面包含了多个变量或数组&#xff0c;它们的类型可以相同&#xff0c;也可以不同&#xff0c;每个这样的变量或数组都称为结构体的成员&#xff…

ESP32学习笔记20-dac

20.DAC 20.1概述 ESP32 有两个 8 位数模转换器 (DAC) 通道,分别连接到 GPIO25(通道 1)和 GPIO26(通道 2)每个 DAC 通道可以将数字值 0-255 转换成模拟电压 0-Vrefout_voltage = Vref * digi_val / 255DAC 外设支持以下列方式输出模拟信号: 直接输出电压。DAC 通道持续输…

代码命名规范是真优雅呀!代码如诗

日常编码中&#xff0c;代码的命名是个大的学问。能快速的看懂开源软件的代码结构和意图&#xff0c;也是一项必备的能力。那它们有什么规律呢&#xff1f; Java项目的代码结构&#xff0c;能够体现它的设计理念。Java采用长命名的方式来规范类的命名&#xff0c;能够自己表达…

Springboot常用注解总结

目录 一、什么是Spring Boot二、Spring常用注解三、Spring Boot常用注解1、SpringBootApplication2、ImportAutoConfiguration3、SpringBootConfiguration4、ImportResource5、PropertySource6、PropertySources7、Role8、Scope9、Lazy11、Profile12、DependsOn13、PostConstru…