面向对象程序设计——set容器の简析

news/2024/9/23 1:32:10/

1.set的介绍

• 序列式容器和关联式容器

• 我们已经接触过STL中的部分容器如:string、vector、list、deque、array、forward_list等,这些容器统称为序列式容器,因为逻辑结构为线性序列的数据结构,两个位置存储的值之间⼀般没有紧密的关联关系,⽐如交换⼀下,他依旧是序列式容器。

• 顺序容器中的元素是按他们在容器中的存储位置来顺序保存和访问的。

• 关联式容器也是⽤来存储数据的,与序列式容器不同的是,关联式容器逻辑结构通常是⾮线性结构, 两个位置有紧密的关联关系,交换⼀下,他的存储结构就被破坏了。顺序容器中的元素是按关键字来保存和访问的。关联式容器有map/set系列和unordered_map/unordered_set系列。set底层是红⿊树,红⿊树是⼀颗平衡⼆叉搜索树。set是key搜索场景的结构

• set的声明如下:

T就是set底层关键字的类型 • set默认要求T⽀持⼩于⽐较,如果不⽀持或者想按⾃⼰的需求⾛可以⾃⾏实现仿函数传给第⼆个模版参数

• set底层存储数据的内存是从空间配置器申请的,如果需要可以⾃⼰实现内存池,传给第三个参数

• ⼀般情况下,我们都不需要传后两个模版参数

• set底层是⽤红⿊树实现,增删查效率是,迭代器遍历是⾛的搜索树的中序,所以是有序的 

2.set的增删查 

2.1插入(insert)

插入接口的介绍 

// 单个数据插⼊,如果已经存在则插⼊失败 
pair<iterator,bool> insert (const value_type& val);
// 列表插⼊,已经在容器中存在的值不会插⼊ 
void insert (initializer_list<value_type> il);
// 迭代器区间插⼊,已经在容器中存在的值不会插⼊ 
template <class InputIterator>
void insert (InputIterator first, InputIterator last);

插入接口的使用 

set容器在插入后默认是升序存储的,并且无法插入重复的元素,但是可以使用仿函数来传参实现降序,也可以插入一段元素,通常使用ASCII码来进行比较大小

//插入
int main()
{//去重+默认升序排列set<int> s1;s1.insert(2);s1.insert(4);s1.insert(3);s1.insert(2);s1.insert(5);s1.insert(9);//迭代器遍历//set<int, greater<int>>::iterator it = s1.begin();auto it_s1 = s1.begin();while (it_s1 != s1.end()){//*it = 1;---->error//不可以修改cout << *it_s1 << " ";++it_s1;}cout << endl;//去重+使用仿函数降序排列set<int,greater<int>> s2;s2.insert(2);s2.insert(4);s2.insert(3);s2.insert(2);s2.insert(5);s2.insert(9);//迭代器遍历//set<int, greater<int>>::iterator it = s2.begin();auto it_s2 = s2.begin();while (it_s2 != s2.end()){cout << *it_s2 << " ";++it_s2;}cout << endl;//插⼊⼀段initializer_list列表值,已经存在的值插⼊失败set<int> s3;s3.insert({ 2,3,4,5,5,7,9,11 });for (auto e : s3){cout << e << " ";}cout << endl;//插入string类,通过ASCII码的大小来排序set<string> s4;s4.insert({ "zhangsan", "lisi", "wangwu" });for (auto e : s4){cout << e << " ";}cout << endl;return 0;
}

2.2查找(find)

算法库中的find使用的是遍历查找,时间复杂度是O(N)

set自身的find是符合平衡二叉树的查找,时间复杂度O(logN)

// 算法库的查找 O(N) auto pos1 = find(s.begin(), s.end(), x); // set⾃⾝实现的查找 O(logN) auto pos2 = s.find(x); 

查找接口的介绍 

// 查找val,返回val所在的迭代器,没有找到返回end() 
iterator find (const value_type& val);
// 查找val,返回Val的个数 
size_type count (const value_type& val) const;

查找接口的使用 

1. find接口直接删除

int main()
{//去重+默认升序排列set<int> s;s.insert(2);s.insert(4);s.insert(3);s.insert(2);s.insert(5);s.insert(9);//迭代器遍历//set<int, greater<int>>::iterator it = s.begin();auto it = s.begin();while (it != s.end()){//*it = 1;---->error//不可以修改cout << *it << " ";++it;}cout << endl;int x = 0;cout << "请输入你要查找的数字:";cin >> x;//如果查找不到就会返回迭代器的尾部/*auto pos = s.find(x);if (pos != s.end()){cout << x << "存在" << endl;}else{cout << x << "不存在" << endl;}*/return 0;
}

count函数间接查找 

//使用count来间接查找//count可以统计元素出现的次数,如果出现0次就不存在,反之则存在if (s.count(x)){cout << x << "存在" << endl;}else{cout << x << "不存在" << endl;}

2.3删除(erase)

删除接口的介绍 

// 删除⼀个迭代器位置的值 
iterator erase (const_iterator position);
// 删除val,val不存在返回0,存在返回1 
size_type erase (const value_type& val);
// 删除⼀段迭代器区间的值 
iterator erase (const_iterator first, const_iterator last);

有关查找区间的迭代器

首先了解一个概念就是迭代器区间通常都是左闭右开的一个范围区间,也就说[a,b)的一个类型,这里的lower_bound与upper_bound可以实现查找一个左闭右开的区间以供操作 

// 返回⼤于等val位置的迭代器 
iterator lower_bound (const value_type& val) const;
// 返回⼤于val位置的迭代器 
iterator upper_bound (const value_type& val) const;

删除接口的使用 

直接删除并判断是否删除成功 

//删除
int main()
{set<int> s;s.insert({ 2,4,5,2,6,8,10,15 });for (auto e : s){cout << e << " ";}cout << endl;//删除最小的元素就删除排序后的首元素s.erase(s.begin());for (auto e : s){cout << e << " ";}cout << endl;//指定删除元素并判断是否删除成功//可以使用erase的返回值统计待删除元素出现的次数来判断是否删除成功int x = 0;cout << "输入你要删除的元素:";cin >> x;int num = s.erase(x);if (num){cout << "删除成功" << endl;for (auto e : s){cout << e << " ";}cout << endl;}else{cout << "删除失败" << endl;for (auto e : s){cout << e << " ";}cout << endl;}return 0;
}

迭代器删除  

//删除
int main()
{set<int> s;s.insert({ 2,4,5,2,6,8,10,15 });for (auto e : s){cout << e << " ";}cout << endl;//指定删除元素并判断是否删除成功//可以使用erase的返回值统计待删除元素出现的次数来判断是否删除成功int x = 0;cout << "输入你要删除的元素:";cin >> x;//使用迭代器删除//如果未查找到则直接返回迭代器尾部auto pos = s.find(x);if (pos != s.end()){s.erase(x);cout << "删除成功" << endl;for (auto e : s){cout << e << " ";}cout << endl;}else{cout << "删除失败" << endl;for (auto e : s){cout << e << " ";}cout << endl;}return 0;
}

迭代器区间删除 

//迭代器区间删除
int main()
{set<int> s;for (int i = 0; i < 10; i++){s.insert(i * 10);}for (auto e : s){cout << e << " ";}cout << endl;//删除30到60区间的数据//取出 >=30 的迭代器指针,包括30auto low = s.lower_bound(30);//取出 >60 的迭代器指针,不包括60auto up = s.upper_bound(60);s.erase(low, up);for (auto e : s){cout << e << " ";}cout << endl;return 0;
}

3.multiset和set的差异 

multiset和set的使⽤基本完全类似,主要区别点在于multiset⽀持值冗余,那么 insert/find/count/erase都围绕着⽀持值冗余有所差异

小tips:中序第一个指的就是从根节点开始以左子树->根节点->右子树的顺序,当在左子树找到符合的值后继续从该值的左子树寻找,直到找不到为止,这时的节点就是中序的第一个

#include<iostream>
#include<set>
using namespace std;
int main()
{// 相⽐set不同的是,multiset是排序,但是不去重 multiset<int> s = { 4,2,7,2,4,8,4,5,4,9 };auto it = s.begin();while (it != s.end()){cout << *it << " ";++it;}cout << endl;// 相⽐set不同的是,x可能会存在多个,find查找中序的第⼀个 int x;cin >> x;auto pos = s.find(x);while (pos != s.end() && *pos == x){cout << *pos << " ";++pos;}cout << endl;// 相⽐set不同的是,count会返回x的实际个数 cout << s.count(x) << endl;// 相⽐set不同的是,erase给值时会删除所有的x s.erase(x);for (auto e : s){cout << e << " ";}cout << endl;return 0;
}

4.代码练习题 

4.1两个数组的交集

题目来源:349.两个数组的交集 ,这里使用set容器充当去重与排序的作用

class Solution {
public:vector<int> intersection(vector<int>& nums1, vector<int>& nums2) {vector<int> v;set<int> s1;set<int> s2;s1.insert(nums1.begin(),nums1.end());s2.insert(nums2.begin(),nums2.end());auto it1 = s1.begin();auto it2 = s2.begin();while(it1 != s1.end() && it2 != s2.end()){if(*it1 < *it2){it1++;}else if(*it1 > *it2){it2++;}else{v.push_back(*it1);it1++;it2++;}}return v;}
};

4.2环形链表II

题目来源:142.环形链表|| ,这里使用set存储的是链表每个节点,当插入到重复的节点时,该节点就是入环的第一个节点,此时直接返回即可

/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode(int x) : val(x), next(NULL) {}* };*/
class Solution {
public:ListNode *detectCycle(ListNode *head){set<ListNode*> L;ListNode* cur = head;while(cur){if(L.count(cur)){return cur;}else{L.insert(cur);}cur = cur->next;}return nullptr;}
};

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

相关文章

LLM应用实战: 文档问答系统Kotaemon-1. 简介及部署实践

1.背景 本qiang~这两周关注到一个很火的开源文档问答系统Kotaemon&#xff0c;从8月28日至今短短两周时间&#xff0c;github星标迅猛增长10K&#xff0c;因此计划深挖一下其中的原理及奥秘。 本篇主要是Kotaemon的简介信息&#xff0c;涉及到主要特点&#xff0c;与传统文档…

MQ(RabbitMQ)笔记

初识MQ 同步调用优缺点 异步调用优缺点 总结&#xff1a; 时效性要求高&#xff0c;需要立刻得到结果进行处理--->同步调用 对调用结果不关心&#xff0c;对性能要求高&#xff0c;响应时间短--->异步调用

Oracle 数据库安装和配置教程

Oracle 数据库是全球领先的企业级数据库解决方案&#xff0c;广泛应用于各种规模的组织。随着 Oracle 版本的更新&#xff0c;安装和配置步骤也逐渐演变。本文将结合最新资料&#xff0c;详细讲解 Oracle 数据库的安装与配置流程&#xff0c;帮助开发者和数据库管理员顺利部署这…

Excel常见操作命令~你值得拥有!

在Excel中&#xff0c;常用的快捷键及其作用对于提高工作效率至关重要。以下是一些常用的Excel快捷键及其详细说明&#xff1a; 1. 基本操作快捷键 快捷键作用Ctrl C复制选定的单元格或区域Ctrl V粘贴复制的内容到选定位置Ctrl X剪切选定的单元格或区域Ctrl S保存当前工作…

vue.config.js devServer中changeOrigin的作用

问题 vue开发时&#xff0c;为了解决前端跨域问题&#xff0c;通常在vue.config.js配置 devServer proxy devServer: {proxy:{/api: {target: http://b.com,changeOrigin: false},}, }官方文档http-proxy options对changeOrigin的解释 option.changeOrigin: true/false, Defa…

【d46】【Java】【力扣】234.回文链表

思路 判断是否是回文&#xff0c;需要&#xff1a;一个指针指向头&#xff0c;一个指针指向尾&#xff0c;两个指针一边向中间靠拢&#xff0c;一边判断数值是否相同 对于单链表&#xff0c;不方便获得pre&#xff0c;如果将节点放进 数组/list &#xff0c;数组/list可以直接…

《深度学习》PyTorch 常用损失函数原理、用法解析

目录 一、常用损失函数 1、CrossEntropyLoss&#xff08;交叉熵损失&#xff09; 1&#xff09;原理 2&#xff09;流程 3&#xff09;用法示例 2、L1Loss&#xff08;L1损失/平均绝对误差&#xff09; 1&#xff09;原理 2&#xff09;用法示例 3、NLLLoss&#xff08;负对…

【STM32系统】基于STM32设计的DAC输出电压与ADC检测电压系统(简易万用表,检测电压电流)——文末工程资料下载

基于STM32设计的DAC输出电压与ADC检测电压系统(简易万用表,检测电压电流) 演示视频: 基于STM32设计的DAC输出电压与ADC检测电压系统(简易万用表,检测电压电流) 前言:本项目实现对STM32的DAC和ADC的程序设计与硬件电路连接实现STM32内部DAC输出电压,并且ADC可以采集电压…