当前位置: 首页 > news >正文

十四天学会C++之第七天:STL(标准模板库)

1. STL容器

  • 什么是STL容器,为什么使用它们。
  • 向量(vector):使用向量存储数据。
  • 列表(list):使用列表实现双向链表。
  • 映射(map):使用映射实现键值对存储。

什么是STL容器?

STL容器是一种用于存储和管理数据的C++模板类,它们提供了多种数据结构,以满足不同的需求。STL容器分为序列容器(Sequence Containers)和关联容器(Associative Containers)两大类。序列容器类似于数组,关联容器则基于键值对存储数据。

向量(vector)

向量是序列容器中最常用的之一,它类似于动态数组。它具有自动扩展和收缩的能力,因此可以根据需要存储任意数量的元素。

#include <vector>
#include <iostream>int main() {std::vector<int> myVector;  // 创建一个整数向量// 向向量中添加元素myVector.push_back(1);myVector.push_back(2);myVector.push_back(3);// 遍历并输出向量中的元素for (int i : myVector) {std::cout << i << " ";}return 0;
}

列表(list)

列表是双向链表的实现,提供了高效的插入和删除操作。与向量不同,列表的元素在内存中不是连续存储的,因此适用于需要频繁插入和删除操作的情况。列表的简单示例:

#include <list>
#include <iostream>int main() {std::list<std::string> myList;  // 创建一个字符串列表// 向列表中添加元素myList.push_back("Hello");myList.push_back("World");myList.push_front("C++");// 遍历并输出列表中的元素for (const std::string& str : myList) {std::cout << str << " ";}return 0;
}

映射(map)

映射是关联容器,它将键与值关联起来,允许通过键快速查找对应的值。这在需要构建字典、查找表或关系映射时非常有用。以下是一个映射的简单示例:

#include <map>
#include <iostream>int main() {std::map<std::string, int> myMap;  // 创建一个字符串到整数的映射// 向映射中添加键值对myMap["apple"] = 5;myMap["banana"] = 3;myMap["cherry"] = 8;// 查找并输出映射中的值std::cout << "The number of bananas is: " << myMap["banana"] << std::endl;return 0;
}

STL容器的强大功能使它们成为C++开发中不可或缺的一部分。无论你的项目规模如何,STL都提供了各种工具,用于有效地管理和操作数据,让代码更加高效和可维护。

2. STL算法

  • 什么是STL算法,它们的作用。
  • 排序算法:介绍STL的排序算法。
  • 查找算法:介绍STL的查找算法。
  • 使用STL算法来处理容器中的数据。

什么是STL算法?

STL算法是一组通用的、可复用的算法,它们独立于特定的容器类型,可以用于处理各种数据结构。STL算法可以显著提高代码的可读性和可维护性,同时还能提供高效的性能。

排序算法

STL提供了多种排序算法,其中最常用的是std::sort()函数,它可以对容器中的元素进行升序排序。排序算法:

#include <algorithm>
#include <vector>
#include <iostream>int main() {std::vector<int> numbers = {5, 2, 8, 1, 9};// 使用std::sort()对向量进行升序排序std::sort(numbers.begin(), numbers.end());// 输出排序后的结果for (int num : numbers) {std::cout << num << " ";}return 0;
}

查找算法

STL还提供了多种查找算法,其中包括std::find()函数,它可以在容器中查找指定的元素。

#include <algorithm>
#include <vector>
#include <iostream>int main() {std::vector<int> numbers = {5, 2, 8, 1, 9};int target = 8;// 使用std::find()在向量中查找目标元素auto result = std::find(numbers.begin(), numbers.end(), target);// 检查查找结果if (result != numbers.end()) {std::cout << "Found " << target << " at position " << (result - numbers.begin()) << std::endl;} else {std::cout << target << " not found." << std::endl;}return 0;
}

使用STL算法处理容器中的数据

STL算法通常以迭代器作为参数,因此它们可以应用于各种容器,如向量、列表、映射等。这使得算法非常通用和灵活。你可以使用std::for_each()std::transform()等算法来遍历容器中的元素,执行自定义的操作。

3. 自定义模板

  • 什么是模板,为什么使用模板。
  • 函数模板:创建可以处理不同数据类型的通用函数。
  • 类模板:创建可以处理不同数据类型的通用类。

什么是模板?

模板是一种通用的代码蓝图,它允许你编写可以处理不同数据类型的代码。C++中有两种主要类型的模板:函数模板和类模板。

函数模板

函数模板允许你编写通用的函数,这些函数可以接受不同类型的参数。函数模板的定义以template关键字开始,后跟模板参数列表和函数声明。

简单的函数模板,用于交换两个值:

template <typename T>
void swapValues(T &a, T &b) {T temp = a;a = b;b = temp;
}int main() {int x = 5, y = 10;double a = 3.14, b = 2.71;swapValues(x, y);  // 交换整数swapValues(a, b);  // 交换双精度浮点数return 0;
}

函数模板的typename T部分表示模板参数,它可以是任何数据类型。

类模板

类模板允许你编写通用的类,这些类可以处理不同类型的成员变量和成员函数。类模板的定义以template关键字开始,后跟模板参数列表和类定义。

类模板,用于创建通用的栈数据结构:

template <typename T>
class Stack {
public:Stack() : top(-1) {}void push(T item) {if (top < MaxSize - 1) {data[++top] = item;}}T pop() {if (top >= 0) {return data[top--];} else {// 处理栈为空的情况throw std::runtime_error("Stack is empty");}}private:static const int MaxSize = 100;T data[MaxSize];int top;
};int main() {Stack<int> intStack;intStack.push(5);intStack.push(10);int x = intStack.pop();  // 弹出值为10的元素Stack<double> doubleStack;doubleStack.push(3.14);doubleStack.push(2.71);double y = doubleStack.pop();  // 弹出值为2.71的元素return 0;
}

类模板的typename T部分表示模板参数,可以是任何数据类型。

为什么使用模板?

使用模板可以使代码更加通用和灵活,无需为不同的数据类型编写重复的代码。模板还有助于提高代码的可维护性,只需编写一次通用代码,即可适用于多种数据类型。

4. 示例和练习

使用STL容器

使用STL容器的示例代码,演示向量(vector)的用法,包括添加元素、遍历容器和查找元素:

#include <iostream>
#include <vector>int main() {std::vector<int> numbers;  // 创建整数向量// 向向量添加元素numbers.push_back(1);numbers.push_back(2);numbers.push_back(3);// 遍历向量并输出元素for (int num : numbers) {std::cout << num << " ";}// 查找元素是否存在int target = 2;auto it = std::find(numbers.begin(), numbers.end(), target);if (it != numbers.end()) {std::cout << "\n" << target << " found at position " << std::distance(numbers.begin(), it);} else {std::cout << "\n" << target << " not found";}return 0;
}

使用STL算法

使用STL算法的示例代码,演示排序和查找算法的用法:

#include <iostream>
#include <vector>
#include <algorithm>int main() {std::vector<int> numbers = {3, 1, 4, 1, 5, 9, 2, 6, 5, 3};// 排序向量std::sort(numbers.begin(), numbers.end());// 输出排序后的向量for (int num : numbers) {std::cout << num << " ";}// 使用二分查找查找元素int target = 5;if (std::binary_search(numbers.begin(), numbers.end(), target)) {std::cout << "\n" << target << " found";} else {std::cout << "\n" << target << " not found";}return 0;
}

使用模板

使用函数模板的示例代码,如何创建一个通用的函数来比较两个值:

#include <iostream>template <typename T>
bool isEqual(T a, T b) {return a == b;
}int main() {int x = 5, y = 5;double a = 3.14, b = 2.71;if (isEqual(x, y)) {std::cout << "x and y are equal\n";} else {std::cout << "x and y are not equal\n";}if (isEqual(a, b)) {std::cout << "a and b are equal\n";} else {std::cout << "a and b are not equal\n";}return 0;
}

练习题

创建一个向量,存储一组学生的分数,并计算平均分。

#include <iostream>
#include <vector>int main() {std::vector<double> scores; // 创建一个双精度浮点数向量来存储学生的分数// 向向量添加学生分数scores.push_back(85.5);scores.push_back(92.0);scores.push_back(78.5);scores.push_back(88.0);scores.push_back(90.5);// 计算平均分double sum = 0.0;for (double score : scores) {sum += score;}double average = sum / scores.size();// 输出平均分std::cout << "平均分: " << average << std::endl;return 0;
}

运行结果:

在这里插入图片描述

解答:

  • 首先,包含 <iostream><vector> 头文件来使用C++的标准输入输出和向量容器。
  • 然后,创建一个std::vector<double>类型的向量,用于存储学生的分数。
  • 使用 push_back 函数将分数添加到向量中。
  • 通过迭代向量中的分数,计算它们的总和并除以向量的大小来获得平均分。
  • 最后,输出平均分。

使用STL的列表容器存储一组单词,并按字母顺序排序。

#include <iostream>
#include <list>
#include <algorithm>int main() {std::list<std::string> words; // 创建一个字符串列表来存储单词// 向列表添加单词words.push_back("apple");words.push_back("banana");words.push_back("cherry");words.push_back("date");words.push_back("fig");// 使用STL的排序算法对单词进行排序words.sort();// 遍历并输出排序后的单词for (const std::string& word : words) {std::cout << word << " ";}return 0;
}

运行结果:

在这里插入图片描述

解答:

  • 包含了 <iostream><list> 头文件,以便使用C++的标准输入输出和列表容器。
  • 然后,创建一个 std::list<std::string> 类型的列表,用于存储单词。
  • 使用 push_back 函数将单词添加到列表中。
  • 使用 sort 函数对列表中的单词进行字母顺序排序。
  • 最后,遍历并输出排序后的单词。

创建一个通用的函数模板,用于计算两个数的最大值。

#include <iostream>template <typename T>
T findMax(T a, T b) {return (a > b) ? a : b;
}int main() {int intMax = findMax(42, 17);double doubleMax = findMax(3.14, 2.71);std::cout << "最大整数值: " << intMax << std::endl;std::cout << "最大双精度浮点数值: " << doubleMax << std::endl;return 0;
}

运行结果:

在这里插入图片描述

解答:

  • 定义一个通用函数模板 findMax,它可以接受任何类型的参数 ab
  • 函数模板的返回类型也是参数的类型。
  • main 函数中,演示如何使用这个模板函数来查找整数和双精度浮点数的最大值。

创建一个类模板,表示二维坐标点,包括x和y坐标。

#include <iostream>template <typename T>
class Point {
public:Point(T x, T y) : x_(x), y_(y) {}void display() const {std::cout << "X坐标: " << x_ << ", Y坐标: " << y_ << std::endl;}private:T x_;T y_;
};int main() {Point<int> intPoint(2, 3);Point<double> doublePoint(1.5, 2.0);std::cout << "整数坐标点:" << std::endl;intPoint.display();std::cout << "双精度浮点数坐标点:" << std::endl;doublePoint.display();return 0;
}

运行结果:

在这里插入图片描述

解答:

  • 定义一个类模板 Point,它有两个模板参数 typename T,表示x和y坐标的类型。
  • 类模板有一个构造函数,可以接受两个参数,分别用于初始化x和y坐标。
  • 类模板还有一个成员函数 display,用于显示坐标点的值。
  • main 函数中,我们演示了如何创建不同类型的坐标点。

使用STL的映射容器存储学生的姓名和分数,然后按姓名查找分数。

#include <iostream>
#include <map>
#include <string>int main() {std::map<std::string, double> studentScores; // 创建一个映射容器,姓名映射到分数// 向映射容器添加学生和分数studentScores["Alice"] = 90.5;studentScores["Bob"] = 88.0;studentScores["Charlie"] = 92.5;studentScores["David"] = 85.0;// 按姓名查找分数std::string nameToFind = "Bob";if (studentScores.find(nameToFind) != studentScores.end()) {double score = studentScores[nameToFind];std::cout << nameToFind << " 的分数是 " << score << std::endl;} else {std::cout << "找不到学生:" << nameToFind << std::endl;}return 0;
}

运行结果:

在这里插入图片描述

解答:

  • 使用 std::map 容器来创建一个学生姓名到分数的映射。
  • 使用方括号操作符 [] 添加学生和分数的映射。
  • 使用 find 函数按姓名查找分数,如果找到,就输出分数;如果找不到,就输出找不到的消息。

相关文章:

十四天学会C++之第七天:STL(标准模板库)

1. STL容器 什么是STL容器&#xff0c;为什么使用它们。向量&#xff08;vector&#xff09;&#xff1a;使用向量存储数据。列表&#xff08;list&#xff09;&#xff1a;使用列表实现双向链表。映射&#xff08;map&#xff09;&#xff1a;使用映射实现键值对存储。 什么…...

Linux 下安装 miniconda,管理 Python 多环境

安装 miniconda 1、下载安装包 Miniconda3-py37_22.11.1-1-Linux-x86_64.sh&#xff0c;或者自行选择版本 2、把安装包上传到服务器上&#xff0c;这里放在 /home/software 3、安装 bash Miniconda3-py37_22.11.1-1-Linux-x86_64.sh 4、按回车 Welcome to Miniconda3 py37…...

Django和jQuery,实现Ajax表格数据分页展示

1.需求描述 当存在重新请求接口才能返回数据的功能时&#xff0c;若页面的内容很长&#xff0c;每次点击一个功能&#xff0c;页面又回到了顶部&#xff0c;对于用户的体验感不太友好&#xff0c;我们希望当用户点击这类的功能时&#xff0c;能直接加载到数据&#xff0c;请求…...

k8s认证

1. 证书介绍 服务端保留公钥和私钥&#xff0c;客户端使用root CA认证服务端的公钥 一共有多少证书&#xff1a; *Etcd&#xff1a; Etcd对外提供服务&#xff0c;要有一套etcd server证书Etcd各节点之间进行通信&#xff0c;要有一套etcd peer证书Kube-APIserver访问Etcd&a…...

基于python开发的IP修改工具

工作中调试设备需要经常修改电脑IP&#xff0c;非常麻烦&#xff0c;这里使用Pythontkinter做了一个IP修改工具 说明: 1.启动程序读取config.json文件2.如果没有该文件则创建&#xff0c;写入当前网卡信息3.通过配置信息进行网卡状态修改4.更新文件状态&#xff0c;删除或修…...

Mybatis源码分析

1. Mybatis整体三层设计 SSM中&#xff0c;Spring、SpringMVC已经在前面文章源码分析总结过了&#xff0c;Mybatis源码相对Spring和SpringMVC而言是的简单的&#xff0c;只有一个项目&#xff0c;项目下分了很多包。从宏观上了解Mybatis的整体框架分为三层&#xff0c;分别是基…...

python树结构包treelib入门及其计算应用

树是计算机科学中重要的数据结构。例如决策树等机器学习算法设计、文件系统索引等。创建treelib包是为了在Python中提供树数据结构的有效实现。 Treelib的主要特点包括&#xff1a; 节点搜索的高效操作。支持常见的树操作&#xff0c;如遍历、插入、删除、节点移动、浅/深复制…...

Rust之自动化测试(三): 测试组合

开发环境 Windows 10Rust 1.73.0 VS Code 1.83.1 项目工程 这里继续沿用上次工程rust-demo 测试组合 正如本章开始时提到的&#xff0c;测试是一个复杂的学科&#xff0c;不同的人使用不同的术语和组织。Rust社区根据两个主要类别来考虑测试:单元测试和集成测试。单元测试很…...

专业管理菜单的增删改、查重

1&#xff0c;点击专业管理菜单------查询所有专业信息列表 ①点击菜单&#xff0c;切换专业组件 ②切换到列表组件后&#xff0c;向后端发送请求到Servlet ③调用DAO层&#xff0c;查询数据库&#xff08;sql&#xff09;&#xff0c;封装查询到的内容 ④从后端向前端做出…...

vue3插件开发,上传npm

创建插件 在vue3工程下&#xff0c;创建组件vue页: toolset.vue。并设置组件名称。注册全局组件。新建index.js文件。内容如下&#xff0c;可在main.js中引入index.js&#xff0c;注册该组件进行测试。![在这里插入图片描述](https://img-blog.csdnimg.cn/a3409d2cbeec41c797d5…...

python【多线程、单线程、异步编程】三个版本--在爬虫中的应用

并发编程在爬虫中的应用 之前的课程&#xff0c;我们已经为大家介绍了 Python 中的多线程、多进程和异步编程&#xff0c;通过这三种手段&#xff0c;我们可以实现并发或并行编程&#xff0c;这一方面可以加速代码的执行&#xff0c;另一方面也可以带来更好的用户体验。爬虫程…...

大模型LLM相关面试题整理-位置编码-tokenizer-激活函数-layernorm

10 LLMs 位置编码篇 10.1.1 什么是位置编码&#xff1f; 位置编码是一种用于在序列数据中为每个位置添加位置信息的技术。在自然语言处理中&#xff0c;位置编码通常用于处理文本序列。由于传统的神经网络无法直接捕捉输入序列中的位置信息&#xff0c;位置编码的引入可以帮助…...

python在nacos注册微服务

安装 首先需要安装python的nacos sdk pip install nacos-sdk-python 注册 注册过程非常简单&#xff0c;需要注意的是&#xff0c;注册完要定时发送心跳&#xff0c;否则服务会被nacos删掉。 import nacos import timeSERVER_ADDRESSES "http://1.2.3.4:8848" …...

tuxera ntfs2024破解版mac电脑磁盘读写软件

大家都知道由于操作系统的原因&#xff0c;在苹果电脑上不能够读写NTFS磁盘&#xff0c;但是&#xff0c;今天小编带来的这款tuxera ntfs 2024 mac版&#xff0c;完美的解决了这个问题。这是一款在macOS平台上使用的磁盘读写软件&#xff0c;能够实现苹果Mac OS X系统读写Micro…...

【源码】C++坦克大战源码

文章目录 题目介绍你收到的所有文件源码效果展示报告内容 题目介绍 代码量&#xff1a;1450 语言&#xff1a;C 你收到的所有文件 其中一个是devc版本&#xff0c;也可以用visual stdio 运行。 源码效果展示 typedef struct //这里的出现次序指的是一个AI_ta…...

AIO开放接口平台免费畅享ChatGPT聊天、联网互动、学术等服务!更有DALL·E 3最强AI绘图功能!

免费畅享&#xff01; AIO平台ChatGPT联网、聊天、学术等服务&#xff01; AIO开放接口平台 | 服务介绍 ALL IN ONE &#xff08;AIO&#xff09;API服务是LLM(大语言模型)开放接口平台&#xff1a;持续接入各种主流的大模型接口&#xff0c;并提供简单、易用、统一的API交互…...

【python】屈小原现在要为学校写校庆贺文(CTGU百年校庆)

题目&#xff1a; """ 题目描述&#xff1a; 屈小原需要为学校的校庆写一篇贺文&#xff0c;共需写下n个字&#xff0c;但他目前只完成了1个字。屈小原可以进行两种操作&#xff1a; 在文档的末尾添加一个字&#xff0c;这样字数就会变为x1。 写下与当前字数相同…...

探索未来的视觉革命:卷积神经网络的崭新时代(二)

&#x1f497;&#x1f497;&#x1f497;欢迎来到我的博客&#xff0c;你将找到有关如何使用技术解决问题的文章&#xff0c;也会找到某个技术的学习路线。无论你是何种职业&#xff0c;我都希望我的博客对你有所帮助。最后不要忘记订阅我的博客以获取最新文章&#xff0c;也欢…...

博客后台模块续更(三)

四、后台模块-动态路由 实现了这个动态路由功能之后&#xff0c;就能在浏览器web页面登录进博客管理后台了 1. 接口分析 后台系统需要能实现不同的用户权限可以看到不同的功能&#xff0c;即左侧的导航栏 请求方式 请求地址 请求头 GET /getRouters 需要token请求头 …...

第十二届蓝桥杯模拟赛第三期

A填空题 问题描述 请问在 1 到 2020 中&#xff0c;有多少个数与 2020 互质&#xff0c;即有多少个数与 2020 的最大公约数为 1。 参考答案 800 public class Main {public static void main(String[] args) {int ans0;for(int i1;i<2020;i) {if(gcd(2020,i)1) {ans;}}…...

[概率论基本概念4]什么是无偏估计

关键词&#xff1a;Unbiased Estimation 一、说明 对于无偏和有偏估计&#xff0c;需要了解其叙事背景&#xff0c;是指整体和抽样的关系&#xff0c;也就是说整体的叙事是从理论角度的&#xff0c;而估计器原理是从实践角度说事&#xff1b;为了表明概率理论&#xff08;不可…...

Java基础之数组(附带Comparator)

文章目录 基础概念可变参数组数组与ListComparator类1,基本概念2,使用Comparator的静态方法&#xff08;Java 8&#xff09;3,常用Comparator方法4,例子 排序与查找数组复制其他 基础概念 int[] anArray new int[10];只有创建对象时才会使用new关键字&#xff0c;所以数组是个…...

【物联网-ModBus-RTU

物联网-ModBus-RTU ■ 优秀博主链接■ ModBus-RTU介绍■&#xff08;1&#xff09;帧结构■&#xff08;2&#xff09;查询功能码 0x03■&#xff08;3&#xff09;修改单个寄存器功能码 0x06■&#xff08;4&#xff09;Modbus RTU 串口收发数据分析 ■ 优秀博主链接 Modbus …...

SpringBoot2.3.1集成Knife4j接口文档

首先要查看项目中pom文件里面有没有swagger和knife4j的依赖&#xff0c;如果有的话删除&#xff0c;加入以下依赖 <!-- swagger --><dependency><groupId>com.github.xiaoymin</groupId><artifactId>knife4j-openapi3-spring-boot-starter</…...

PyTorch 中cumprod函数计算张量沿指定维度的累积乘积详解和代码示例

torch.cumprod 是 PyTorch 中用于 计算张量沿指定维度的累积乘积&#xff08;cumulative product&#xff09; 的函数。 1、函数原型 torch.cumprod(input, dim, *, dtypeNone, outNone) → Tensor参数说明&#xff1a; 参数说明input输入张量dim累积乘积的维度dtype可选&…...

有没有 MariaDB 5.5.56 对应 MySQL CONNECTION_CONTROL 插件

有没有 MariaDB 对应 MySQL CONNECTION_CONTROL 插件 背景 写这篇文章的目的是因为昨晚半夜突然被call起来&#xff0c;有一套系统的mysql数据库启动失败了。尝试了重启服务器也不行。让我协助排查一下问题出在哪。 分析过程 一开始拿到服务器IP地址&#xff0c;就去数据库…...

青少年编程与数学 01-011 系统软件简介 01 MS-DOS操作系统

青少年编程与数学 01-011 系统软件简介 01 MS-DOS操作系统 1. MS-DOS的历史背景1.1 诞生背景1.2 发展历程1.3 与Windows的关系 2. MS-DOS的技术细节2.1 系统架构2.2 启动过程2.3 内存管理2.4 设备驱动程序 3. MS-DOS的用户界面3.1 命令行界面3.2 配置文件 4. MS-DOS的应用程序与…...

构建 MCP 服务器:第 2 部分 — 使用资源模板扩展资源

该图像是使用 AI 图像创建程序创建的。 这个故事是在多位人工智能助手的帮助下写成的。 这是构建MCP 服务器教程&#xff08;共四部分&#xff09;的第二部分。在第一部分中&#xff0c;我们使用基本资源创建了第一个 MCP 服务器。现在&#xff0c;我们将使用资源模板扩展服务…...

第23讲、Odoo18 邮件系统整体架构

目录 Odoo 邮件系统整体架构邮件发送方式邮件模板配置SMTP 邮件服务器配置邮件发送过程开发中常见邮件发送需求常见问题排查提示与最佳实践完整示例&#xff1a;审批通过自动发邮件门户表单自动邮件通知案例邮件队列与异步发送邮件添加附件邮件日志与调试多语言邮件模板邮件安…...

MySQL中的内置函数

文章目录 一、日期函数1.1 获取当前的日期1.2 获取当前时间1.3 获取当前日期和时间1.4 提取时间日期1.5 添加日期1.6 减少日期1.7 两个日期的差值 二、字符串处理函数2.1 获取字符串的长度2.2 获取字符串的字节数2.3 字符串拼接2.4 转小写2.5 转大写2.6 子字符串第⼀次出现的索…...