STL的六大组件

一.总体概念

STL(Standard Template Library,标准模板库)是C++标准库的一部分,提供了丰富且高效的数据结构和算法。STL主要由六大组件组成,它们是:

  1. 容器(Containers):STL提供了多种不同类型的容器,如vector、list、deque、set、map等,用于存储和管理数据。这些容器提供了不同的特性和性能,可以根据具体需求选择合适的容器。

  2. 算法(Algorithms):STL包含了大量的算法,如排序、查找、遍历等,可以在不同的容器上进行操作。这些算法具有高度的重用性和通用性,帮助开发者快速实现各种常见的操作。

  3. 迭代器(Iterators):迭代器是STL中用于遍历容器元素的工具,提供了统一的访问接口,使得算法可以以通用的方式操作各种容器。迭代器分为输入迭代器、输出迭代器、正向迭代器、双向迭代器和随机访问迭代器等不同类型。

  4. 仿函数(Functors):仿函数是STL中的一个重要概念,它允许将函数对象作为参数传递给算法,从而实现更加灵活和通用的操作。STL提供了一些内置的仿函数,同时也支持用户自定义的仿函数。

  5. 适配器(Adapters):适配器可以在已有的容器或算法之间提供一个接口层,以实现不同组件之间的互操作。常见的适配器有stack、queue和priority_queue等,它们提供了不同的数据结构和访问方式。

  6. 分配器(Allocators):分配器是STL中用于内存管理的组件,负责为容器分配和释放内存。STL中的容器在实现时通常使用分配器进行内存分配,不同的分配器可以满足不同的内存分配策略和需求。

这些组件共同构成了STL的核心部分,为C++程序提供了强大、高效且通用的数据结构和算法支持。使用STL可以提高代码的可读性、可维护性和性能,并加速开发过程。

在这里插入图片描述

二.详细分析

1.容器

multimap和multiset介绍

Multimap和Multiset都是关联容器,允许存储多个相同键值的元素,但它们之间存在一些区别。

  1. Multimap(多映射):
  • Multimap是一个有序容器,它使用红黑树(Red-Black Tree)作为底层实现,元素按键值自动排序。
  • 允许存储多个相同键值的元素,可以通过键值快速查找对应的元素。
  • 插入新元素时,并不会覆盖掉已存在的相同键值元素,而是将新元素插入到相同键值元素的后面。
  • 提供了插入、删除、查找等操作,时间复杂度为O(log n)。
  • 示例:
#include <iostream>
#include <map>int main() {std::multimap<int, std::string> myMultimap;myMultimap.insert(std::make_pair(1, "apple"));myMultimap.insert(std::make_pair(2, "banana"));myMultimap.insert(std::make_pair(1, "avocado"));for (const auto& pair : myMultimap) {std::cout << pair.first << ": " << pair.second << std::endl;}return 0;
}

输出:

1: apple
1: avocado
2: banana
  1. Multiset(多集合):

Multiset是一种有序容器,允许存储多个相同的元素,并且会保持元素插入的顺序。在Multiset中,相同的元素会被当作不同的元素存储,即允许存在多个相同的元素。

因此,Multiset允许存储多个相同的元素,并且会按照插入的顺序保持这些元素,但不会有重复的元素。

  • Multiset也是一个有序容器,底层同样使用红黑树进行实现,但存储的是无重复的元素集合。
  • 允许存储多个相同的元素,插入时会保持元素的顺序,但不会有重复的元素。
  • 提供了插入、删除、查找等操作,时间复杂度为O(log n)。
  • 示例:
#include <iostream>
#include <set>int main() {std::multiset<int> myMultiset;myMultiset.insert(1);myMultiset.insert(2);myMultiset.insert(1);for (const auto& value : myMultiset) {std::cout << value << std::endl;}return 0;
}

输出:

1
1
2

总结:Multimap和Multiset在STL中提供了方便的操作多个相同键值的元素和无重复元素集合的功能,适用于需要按照特定顺序存储、查找和操作元素的情况。通过了解它们的区别和特性,可以更好地选择适合的容器来满足编程需求。

2. 算法

merge函数介绍

merge是STL中的一个算法,用于将两个已排好序的序列合并成一个有序序列。该算法会合并两个序列,并将结果存储在目标序列中。以下是merge算法的详细介绍:

  • 函数签名:
template <class InputIterator1, class InputIterator2, class OutputIterator>
OutputIterator merge(InputIterator1 first1, InputIterator1 last1,InputIterator2 first2, InputIterator2 last2,OutputIterator result);
参数:
first1, last1: 第一个有序序列的起始和结束迭代器。
first2, last2: 第二个有序序列的起始和结束迭代器。
result: 目标序列的起始迭代器,用于存储合并后的有序序列。
返回值:
返回一个迭代器,指向合并后的有序序列最后一个元素之后的位置。
功能:
merge算法将两个已排好序的序列进行合并,并将结果存储在目标序列中。
合并后的序列仍然是有序的。
  • 使用示例:
#include <iostream>
#include <algorithm>
#include <vector>int main() {std::vector<int> v1 = {1, 3, 5, 7};std::vector<int> v2 = {2, 4, 6, 8};std::vector<int> result(v1.size() + v2.size());std::merge(v1.begin(), v1.end(), v2.begin(), v2.end(), result.begin());for (const auto& num : result) {std::cout << num << " ";}return 0;
}

输出:

1 2 3 4 5 6 7 8

在上面的示例中,我们使用merge算法将两个已排好序的向量v1和v2合并到result中,并输出合并后的有序序列。这样我们可以方便地将两个有序序列合并成一个有序序列。

3.迭代器

迭代器(Iterator)是C++中用来访问容器元素的一种抽象概念。它允许我们以统一的方式遍历各种不同类型的容器,如数组、向量、链表和映射等。下面是一个简单的示例,用于说明迭代器的基本用法:

#include <iostream>
#include <vector>int main() {std::vector<int> v = {1, 2, 3, 4, 5};// 使用迭代器遍历向量并打印元素std::cout << "Using iterator to traverse the vector: ";for (std::vector<int>::iterator it = v.begin(); it != v.end(); ++it) {std::cout << *it << " ";}std::cout << std::endl;return 0;
}

在上面的例子中,我们创建了一个整数向量v并使用迭代器it来遍历它。现在让我解释一下代码中迭代器的用法和相关概念:

  • v.begin()返回指向向量起始元素的迭代器。
  • v.end()返回指向向量末尾下一个位置的迭代器。
  • it是一个迭代器,它初始化为v.begin(),逐步向前遍历直到v.end(),在每一步中,使用*it来获取迭代器当前所在位置的元素值。

上述代码中,我们使用了模板类std::vector的迭代器std::vector::iterator来遍历整数向量v。需要注意的是,C++ 11引入了auto关键字,可以用它来自动推断迭代器的类型,让代码更加简洁:


#include <iostream>
#include <vector>int main() {std::vector<int> v = {1, 2, 3, 4, 5};// 使用 auto 关键字简化迭代器类型std::cout << "Using iterator to traverse the vector: ";for (auto it = v.begin(); it != v.end(); ++it) {std::cout << *it << " ";}std::cout << std::endl;return 0;
}

在这个简化的示例中,我们使用auto关键字自动推断迭代器的类型,并且不需要显式指定std::vector::iterator。

4. 适配器

在C++中,适配器(Adapter)是一种设计模式,用于将一个类的接口转换成另一个类的接口,以满足客户端的需求而不改变原有类的代码。在STL中,有一些适配器容器,可以对其他容器进行适配,提供不同的接口和功能。

下面是几种常见的适配器容器及其功能:

  1. 栈(stack):
  • 栈是一种后进先出(LIFO)的容器,只能在顶部进行插入和删除操作。
  • 使用适配器std::stack可以创建一个栈,基于其他容器(默认是std::deque)实现。
#include <iostream>
#include <stack>int main() {std::stack<int> s;s.push(10);s.push(20);s.push(30);while (!s.empty()) {std::cout << s.top() << " ";s.pop();}std::cout << std::endl;return 0;
}
输出:30 20 10
  1. 队列(queue):
  • 队列是一种先进先出(FIFO)的容器,支持在队尾插入,在队头删除元素。
  • 使用适配器std::queue可以创建一个队列,基于其他容器(默认是std::deque)实现。
#include <iostream>
#include <queue>int main() {std::queue<int> q;q.push(10);q.push(20);q.push(30);while (!q.empty()) {std::cout << q.front() << " ";q.pop();}std::cout << std::endl;return 0;
}
输出:10 20 30
  1. 优先队列(priority_queue):
  • 优先队列是一种具有优先级顺序的队列,每次取出的元素都是优先级最高的。
  • 使用适配器std::priority_queue可以创建一个优先队列,默认情况下基于std::vector实现。
#include <iostream>
#include <queue>int main() {std::priority_queue<int> pq;pq.push(30);pq.push(10);pq.push(20);while (!pq.empty()) {std::cout << pq.top() << " ";pq.pop();}std::cout << std::endl;return 0;
}
输出:30 20 10

以上是几种常见的STL适配器容器以及它们的基本用法。适配器容器可以简化具体容器的操作,并提供了更高层次的功能和接口,方便在实际开发中使用。

5. 仿函数

在C++中,仿函数(function object)是一种类或结构体,可以像函数一样被调用。仿函数可以以类似函数调用的方式使用,将其对象当作函数来调用,这种方式在STL中被广泛应用。

仿函数的特点:

  • 仿函数是一个类或结构体,重载了operator()函数。
  • 仿函数可以带有状态,可以在构造函数中初始化,因此可以用于实现更复杂的逻辑。
  • 仿函数可以作为参数传递给算法或者容器,以自定义比较、排序、筛选等操作。

下面是一个简单的示例,演示如何定义一个仿函数并使用它:

#include <iostream>
#include <algorithm>
#include <vector>// 定义一个仿函数 LessThan,用于比较两个整数的大小
class LessThan {
public:bool operator()(int a, int b) const {return a < b;}
};int main() {std::vector<int> nums = {5, 2, 8, 1, 6, 3};// 使用仿函数 LessThan 对 nums 中的元素进行排序std::sort(nums.begin(), nums.end(), LessThan());// 输出排序后的结果for (int num : nums) {std::cout << num << " ";}std::cout << std::endl;return 0;
}

在这个示例中,我们定义了一个仿函数LessThan,其中重载了operator()函数,用于比较两个整数的大小。然后在main函数中,使用LessThan仿函数作为std::sort算法的第三个参数,实现对nums向量中元素的排序。

在实际开发中,仿函数非常灵活,可以根据需求自定义功能,例如自定义排序条件、筛选条件等。通过仿函数,我们可以更加灵活地使用STL算法和容器,满足不同场景下的需求。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/39119.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

1.(vue3.x+vite)实现卷帘效果

前端技术社区总目录(订阅之前请先查看该博客) 1:效果预览 2:代码编写 <template><div style="width

如何使用Maxscript访问C#类库?

本教程帮助你一步一步集成maxscript文档和C#类库&#xff0c;以便你能够在maxscript提供的相当有限的环境中访问dotnet框架的强大功能。这让你可以使用maxscript做一些功能更强大的事情&#xff0c;比如访问数据库、获取web部署的内容等等。 还是直接上教程实惠&#xff1a; …

昇思25天学习打卡营第6天|关于函数与神经网络梯度相关技术探讨

目录 Python 库及 MindSpore 相关模块和类的导入 函数与计算图 微分函数与梯度计算 Stop Gradient Auxiliary data 神经网络梯度计算 Python 库及 MindSpore 相关模块和类的导入 Python 中的 numpy 库被成功导入&#xff0c;并简称为 np。numpy 在科学计算领域应用广泛&#x…

SQLyog脚本无限试用重置脚本

文章目录 引言脚本(win)必要操作、说明 引言 SQLyog 需要po jie&#xff0c;但是网上的没看到很好使的&#xff0c;直接下的官方。能处理14天试用也是很ok的。 脚本(win) echo offREM SQLyog注册表key&#xff0c;可能跟你的不一样&#xff0c;如果不一样&#xff0c;请替换…

ai绘画一条作品变现1400+,怎么做一个赚钱的AI绘画账号?

大家都知道现在AI很火&#xff0c;变现的玩法也多种多样&#xff0c;但一说到AI&#xff0c;大家就下意识认为这东西离我太远&#xff0c;自己没有那么高学历&#xff0c;不会英文&#xff0c;不会用AI模型等。 其实Ai没有大家想象得那么难&#xff0c;尤其在AI绘画这块&#…

DNS访问百度

DNS&#xff0c;英文全称是 domain name system&#xff0c;域名解析系统&#xff0c;它的作用也很明确&#xff0c;就是域名和 IP 相互映射。 假设你要查询 baidu.com 的 IP 地址: 首先会查找浏览器的缓存,看看是否能找到 baidu.com 对应的IP地址&#xff0c;找到就直接返回&…

【第七节】C/C++排序算法

目录 前言 一、冒泡排序 二、选择排序 三、插入排序 四、希尔排序 五、归并排序 六、快速排序 七、 堆排序 八、计数排序 九、桶排序 十、基数排序 前言 排序算法可以大致分为两大类&#xff1a;比较类排序和非比较类排序。以下是这两大类中一些常见的排序算法示例&…

ChatGPT-4o医学应用、论文撰写、数据分析与可视化、机器学习建模、病例自动化处理、病情分析与诊断支持

2022年11月30日&#xff0c;可能将成为一个改变人类历史的日子——美国人工智能开发机构OpenAI推出了聊天机器人ChatGPT-3.5&#xff0c;将人工智能的发展推向了一个新的高度。2023年11月7日&#xff0c;OpenAI首届开发者大会被称为“科技界的春晚”&#xff0c;吸引了全球广大…

WPF布局控件

目录 Grid StackPanel WrapPanel DockPanel UniformGrid Canvas&InkCanvas Canvas InkCanvas Border Grid 属性 ShowGridLines&#xff1a;显示边线 ColumnDefinitions 列集合 表示有几列下面就写几个ColumnDefinition Width 宽&#xff1a;如果写具体数字则表…

科普文:一文搞懂jvm实战(一)Runtime实时监控jvm

概叙 Java Runtime 类是 Java 标准库中的关键类之一。它提供了对当前Java虚拟机&#xff08;JVM&#xff09;实例的访问和控制&#xff0c;允许程序动态地修改和管理运行时环境。 Java Runtime 是Java虚拟机&#xff08;JVM&#xff09;的一个实例&#xff0c;代表了正在执行Ja…

Spring Boot 实现 AOP 动态热插拔功能并附DEMO源码

&#x1f604; 19年之后由于某些原因断更了三年&#xff0c;23年重新扬帆起航&#xff0c;推出更多优质博文&#xff0c;希望大家多多支持&#xff5e; &#x1f337; 古之立大事者&#xff0c;不惟有超世之才&#xff0c;亦必有坚忍不拔之志 &#x1f390; 个人CSND主页——Mi…

python中的包和模块

目录 一、包与模块 二、第三方包的安装 2.1 pip install 2.2使用 curl 管道 2.3其他安装方法 三、导入单元的构成 3.1pip的使用 四、模块的缓存 一、包与模块 Python 中除了函数库以外&#xff0c;还有非常多且优秀的第三方库、包、模块。 模块Module&#xff1a;以…

【linux网络(七)】数据链路层详解

&#x1f493;博主CSDN主页:杭电码农-NEO&#x1f493;   ⏩专栏分类:Linux从入门到精通⏪   &#x1f69a;代码仓库:NEO的学习日记&#x1f69a;   &#x1f339;关注我&#x1faf5;带你学更多操作系统知识   &#x1f51d;&#x1f51d; Linux网络 1. 前言2. 认识MAC…

【计算机毕业设计】061互助学习微信小程序

&#x1f64a;作者简介&#xff1a;拥有多年开发工作经验&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。&#x1f339;赠送计算机毕业设计600个选题excel文件&#xff0c;帮助大学选题。赠送开题报告模板&#xff…

北京网站建设怎么开始做

北京作为中国的首都&#xff0c;拥有众多的企业和机构&#xff0c;网站建设不仅是一种宣传和推广的手段&#xff0c;更是企业发展的必备工具。但是对于很多企业来说&#xff0c;网站建设是一个相对陌生的领域&#xff0c;不知道从哪里开始。今天我们就来谈一谈北京网站建设的步…

Dockerhub无法拉取镜像配置阿里镜像加速器

打开阿里镜像加速地址&#xff1a; https://cr.console.aliyun.com/cn-hangzhou/instances/mirrors 根据平台类型按照对应方式进行配置&#xff1a;Dokcer Desktop是在右上角点开配置 找到Docker Engine 进行设置JSON结构&#xff1a; 记得要重启Docker服务才会生效&#xff01…

深度学习笔记: 最详尽解释预测系统的分类指标(精确率、召回率和 F1 值)

欢迎收藏Star我的Machine Learning Blog:https://github.com/purepisces/Wenqing-Machine_Learning_Blog。如果收藏star, 有问题可以随时与我交流, 谢谢大家&#xff01; 预测系统的分类指标(精确率、召回率和 F1 值) 简介 让我们来谈谈预测系统的分类指标以及对精确率、召回…

SpringSecurity6 | 基于数据库实现登录认证

SpringSecurity6 | 基于数据库认证 ✅作者简介:大家好,我是Leo,热爱Java后端开发者,一个想要与大家共同进步的男人😉😉 🍎个人主页:Leo的博客 💞当前专栏: 循序渐进学SpringSecurity6 ✨特色专栏: MySQL学习 🥭本文内容: SpringSecurity6 | 基于数据库实现登…

数据资产的价值变现及管理规划(AMT企源)

从数据资源到数据资产之数据资产的价值变现及管理规划 题记 本文旨在探讨数据资产发展途径、数据产权及价值变现的服务流程和路径&#xff0c;并对数据资产管理平台框架、数据资产管理实施规划做出初步解读&#xff0c;以期为数据资产管理提供有益的思路和方案。 本次推出《从…

09 - Python图形用户界面和游戏开发

图形用户界面和游戏开发 基于tkinter模块的GUI GUI是图形用户界面的缩写&#xff0c;图形化的用户界面对使用过计算机的人来说应该都不陌生&#xff0c;在此也无需进行赘述。Python默认的GUI开发模块是tkinter&#xff08;在Python 3以前的版本中名为Tkinter&#xff09;&…