C++模板初识

embedded/2025/2/2 9:14:17/

文章目录

    • 一、函数模板
      • 1. 函数模板
      • 2. 函数模板、模板的特例化、非模板函数的重载关系
      • 3. 外部调用
      • 4. 模板的非类型参数
      • 5. 代码示例
    • 二、类模板

一、函数模板

1. 函数模板

  1. 模板函数:在函数调用点,编译器用用户指定的类型,从原模板实例化一份函数代码出来。

  2. 模板的实参推演 => 可以根据用户传入的实参的类型,来推导出模板类型参数的具体类型。

  3. 函数模板是不进行编译的,因为类型还不知道;
    模板的实例化 - 函数调用点进行实例化模板函数才是要被编译器所编译的。

2. 函数模板、模板的特例化、非模板函数的重载关系

不属于重载关系,属于共存关系

3. 外部调用

模板代码是不能在一个文件中定义的。在另外一个文件中使用的模板代码调用之前,一定要看到模板定义的地方,这样的话,模板才能够进行正常的实例化,产生能够被编译器编译的代码。

	// 告诉编译器,进行指定类型的模板实例化template bool compare<int>(int, int);template bool compare<double>(double, double);

所以,模板代码都是放在头文件当中的,然后在源文件当中直接进行#include包含。

4. 模板的非类型参数

必须是整数类型(整数或地址/引用都可以),都是常量,只能使用,而不能修改。

	template<typename T int SIZE>void sort(T *arr) {for(int i = 0; i < SIZE - 1; ++i)for(int j = 0; j < SIZE - 1 - i; ++j)if(arr[j] > arr[j + 1]) {int tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;}}int arr[] = { 12, 5, 7, 89, 32, 21, 35 };const int size = sizeof(arr) / sizeof(arr[0]);sort<int, size>(arr);

5. 代码示例

#include <iostream>
using namespace std;// 函数模板 
template<typename T> // 定义一个模板参数列表
bool compare(T a, T b) {cout << "template compare" << endl;return a > b;
}// 模板特例化 - 针对compare函数模板,提供const char*类型的特例化版本
template<>
bool compare<const char*>(const char* a, const char* b) {cout << "compare<const char*>" << endl; return strcmp(a, b) > 0;
}// 非模板函数 - 普通函数
bool compare(const char* a, const char* b) {cout << "normal compare" << endl;return strcmp(a, b) > 0;
}int main() {compare<int>(10, 20);compare(100, 20);compare("100", "20");compare<const char*>("100", "20");return 0;
}

输出结果:

template compare
template compare
normal compare
compare<const char*>

二、类模板

#include <iostream>
using namespace std;// 类模板 -> 实例化 -> 模板类
template<typename T>
class SeqStack { // 模板名称 + 类型参数列表 = 类名称
public:// 构造和析构函数名不用加<T>,其它出现模板的地方都加上类型参数列表SeqStack(int size = 10): _pstack(new T[size]), _top(0), _size(size){}~SeqStack() {delete[]_pstack;_pstack = nullptr;}SeqStack(const SeqStack<T>& stack): _top(stack._top), _size(stack._size){_pstack = new T[_size];// 不要用memory进行拷贝for (int i = 0; i < _top; ++i) {_pstack[i] = stack._pstack[i];}}SeqStack<T>& operator=(const SeqStack<T>& stack) {if (this == &stack) {return *this;}delete[]_pstack;_top = stack._top;_size = stack._size;_pstack = new T[_size];// 不要用memory进行拷贝for (int i = 0; i < _top; ++i) {_pstack[i] = stack._pstack[i];}return *this;}//void push(const T& val) { // 类外实现//	if (full())//		expand();//	_pstack[_top++] = val;//}void push(const T& val); void pop() {if (empty())return;--_top;}T top() const {if (empty())throw "stack is empty!"; // 抛异常也代表函数逻辑结束return _pstack[_top - 1]; }bool full() const { return _top == _size; }bool empty() const { return _top == 0; }
private:T *_pstack;int _top;int _size;// 顺序栈底层数组按2倍的方式扩容void expand() {T* ptmp = new T[_size * 2];for (int i = 0; i < _top; ++i) {ptmp[i] = _pstack[i];}delete[]_pstack;_pstack = ptmp;_size *= 2;}
};template<typename T>
void SeqStack<T>::push(const T& val) {if (full())expand();_pstack[_top++] = val;
}int main() {// 类模板的选择性实例化:只有调用到的类模板的函数才会实例化SeqStack<int> s1;s1.push(20);s1.push(30);s1.push(40);s1.push(50);s1.push(60);s1.pop();cout << s1.top() << endl; // 50// 类型参数如有默认值template<typename T=int>//SeqStack<> s2;return 0;
}

http://www.ppmy.cn/embedded/158867.html

相关文章

学习Python编程,需要哪些编程语言基础?如何开始学习Python?

一、编程语言基础 学习Python编程&#xff0c;虽然并不严格要求具备其他编程语言的先验知识&#xff0c;但拥有一些基本的编程概念和理解会对学习过程产生积极影响。以下是一些建议的编程语言基础&#xff0c;它们将帮助你更快地掌握Python&#xff1a; 基本编程概念&#xf…

MySQL知识点总结(十二)

请说明EXPLAIN语句的作用 EXPLAIN语句能够生成SQL语句的执行计划&#xff0c;显示出优化器如何执行该SQL语句的最优路径。可以用来检查SELECT,INSERT,REPLCAE,UPDATE和DELETE语句。 为什么使用索引会比全表扫描提供更好的查询性能&#xff1f; 索引是一个方案对象&#xff0…

c++ list

1.构造函数 构造函数 // list<T> lst; // list(beg, end); // 区间构造 // list(n, elem); // 元素构造 // list(const list &lst); // 拷贝构造#include <iostream> #include <fstream> #include <string> #include <list> using name…

快速启动与休眠唤醒的区分

留着备查&#xff0c;写驱动的方式可以判断真假S4&#xff1b; 链接

G. XOUR

题目链接&#xff1a;Problem - G - Codeforces 题目大意&#xff1a;给你一个n长的序列&#xff0c; 其中你可以将a[i] XOR a[j] 的值 严格小于4的数对进行交换。 你可以操作任何几次&#xff0c; 让最后的数列最小。如果在 x 和 y 不同的第一个位置&#xff0c; xi<yi &…

软件测试 —— jmeter(2)

软件测试 —— jmeter&#xff08;2&#xff09; HTTP默认请求头&#xff08;元件&#xff09;元件作用域和取样器作用域HTTP Cookie管理器同步定时器jmeter插件梯度压测线程组&#xff08;Stepping Thread Group&#xff09;参数解析总结 Response Times over TimeActive Thre…

Flink报错Caused by: java.io.FileNotFoundException: /home/wc.txt

当在提交一个flink任务报如下的错误时&#xff1a; Caused by: java.io.FileNotFoundException: /home/wc.txt (没有那个文件或目录)at java.io.FileInputStream.open0(Native Method)at java.io.FileInputStream.open(FileInputStream.java:195)at java.io.FileInputStream.&…

分布式微服务系统架构第87集:kafka

Kafka 就是为了解决上述问题而设计的一款基于发布与订阅的消息系统。它一般被称为 “分布式提交日志”或者“分布式流平台”。文件系统或数据库提交日志用来提供所有事务 的持久记录&#xff0c;通过重放这些日志可以重建系统的状态。同样地&#xff0c;Kafka 的数据是按照一定…