详解c++---set的介绍

news2024/11/25 16:43:59

目录标题

  • set容器的介绍
  • set的构造函数
  • insert函数的介绍
  • find函数
  • erase函数
  • count函数
  • lower_bound
  • upper_bound
  • multiset

set容器的介绍

set容器可以看成我们上一篇文章学习的K结构的搜索二叉树,所以set容器不仅可以存储数据,还可以对数据进行排序和去重,另外set属于关联式容器也就是说set的每个元素都是通过某种关系相互链接起来的不想vector一样在虚拟地址上一个元素紧接着一个元素,所以set容器种没有push_bck函数只有insert函数,那么接下来我们就来看看insert函数的介绍。
在这里插入图片描述

set的构造函数

在这里插入图片描述
set容器提供了三种构造函数我们首先来看第一种,这种构造函数有两个参数并且每个参数都有一个缺省值,那么第一个参数表示的意思就是搜索二叉树的比较方法,如果库中提供的够用的话这个参数就不需要传,第二个参数就是一个有关内存池的参数,那么这个参数对于我们初学者也不需要管,所以第一种构造函数对于我们来说几乎什么参数都不用传,直接用它创建一个空容器就可以了,比如说下面的代码:

void func()
{
	set<int> tmp1;
	set<long long> tmp2;
	set<char> tmp3;
	set<double> tmp4;
}

那么第二种的形式有两个迭代器参数,那么这种形式表示的意思就是用迭代器范围的内容构造一个容器,比如说下面的代码:

void func()
{
	vector<int> v = { 1,2,3,4,5 };
	set<int> tmp(v.begin(), v.end());
}

那么这种形式创建的容器在一开始就会有数据:
在这里插入图片描述
第三种形式就是拷贝构造函数,这种形式就不必多讲了,大家可以看看下面的代码:

void func()
{
	vector<int> v = { 1,2,3,4,5 };
	set<int> tmp(v.begin(), v.end());
	set<int> copy_tmp(tmp);
}

这里通过调试就可以看到两个容器里面的内容是一样的:
在这里插入图片描述
那么这就是set的构造函数,很简单跟之前学的差不多。

insert函数的介绍

在这里插入图片描述
首先参数中的value_type就是模板参数中的一个参数的重命名,那么知道这个之后我们就可以明白第一个形式的insert函数表示的意思就是往容器中插入一个T类型的数据,比如说下面的代码:

void func1()
{
	set<int> tmp;
	tmp.insert(2);
	tmp.insert(1);
	tmp.insert(3);
	tmp.insert(0);
}

因为名为tmp创建的时候显示实例化的类型是int,所以我们可以通过insert函数向里面插入int类型的数据,第二种形式的参数多了一个迭代器类型的参数,那么这种形式表示的意思就是往容器中的指定位置插入数据,那么这种形式希望大家谨慎使用因为可能会导致树的结构发生破坏,使其失去搜索二叉树的特征。第三种形式的参数就是两个迭代器,那么这种形式的insert函数表示的意思就是将一个范围内的数据插入到容器里面,那么这个迭代器类型可以是任意容器比如说下面的代码:

void func1()
{
	set<int> tmp1;
	vector<int> v = { 1,3,5,7,9 };
	list<int> l = { 2,4,6,8,10 };
	tmp1.insert(v.begin(), v.end());
	tmp1.insert(l.begin(), l.end());
}

那么这里我们可以通过调试查看tmp1中的数据内容:
在这里插入图片描述
那么这就是set的insert函数的第三个形式的作用希望大家能够理解。

find函数

find函数想必大家都应该知道这个函数的作用,那么这里我们首先来看看这个函数的参数形式:
在这里插入图片描述
find函数只有一个形式,这种形式的意思传递你要找的数据就行,如果找到的话这里会返回这个数组所在的容器中的位置,如果找不到的话这里会返回一个指向容器结尾end的迭代器,比如说下面的代码:

void func2()
{
	vector<int> v = { 1,3,5,7,9 };
	set<int> tmp(v.begin(),v.end());
	set<int> ::iterator it = tmp.find(3);
	if (it != tmp.end())
	{
		cout << "找到了数据为:" << *it << endl;
	}
	else
	{
		cout << "容器中没有这个数据" << endl;
	}

}

这段代码的运行结果如下:
在这里插入图片描述
如果我们要找一个不存在的数据的话这里运行的结果如下,比如说把上面的3改成4运行的结果就如下:
在这里插入图片描述
那么这里就有一个问题,set库中提供了find函数,但是算法库中也提供了find函数,并且这个函数都可以达到我们的预期,比如说下面的代码:

void func2()
{
	vector<int> v = { 1,3,5,7,9 };
	set<int> tmp(v.begin(),v.end());
	set<int> ::iterator it = find(tmp.begin(), tmp.end(), 3);
	if (it != tmp.end())
	{
		cout << "找到了数据为:" << *it << endl;
	}
	else
	{
		cout << "容器中没有这个数据" << endl;
	}

}

这段代码的运行结果如下:
在这里插入图片描述

那我们用哪个find函数好呢?能不能为了统一就只是用算法库中的find函数呢?答案是最好不要因为算法库中的find函数是暴露查找通过循环逐个比对迭代器中的值是否和要找的值相匹配,这种查找方式并没有发挥搜索二叉树所特有的性质所以不推荐使用,而set库中的find函数充分的利用了二分查找的性质,所以他的效率会更高,我们更推荐使用,那么这就是两个find函数的区别,虽然都能达到目的,但是set的效率会更高。

erase函数

我们来看看erase函数的参数形式:
在这里插入图片描述

erase函数支持三种参数的删除,第一种的参数就是一个迭代器表示的意思就是将迭代器指向的位置的元素进行删除,比如说下面的代码:

void func3()
{
	vector<int> v = { 2,1,5,4,3 };
	set<int> tmp(v.begin(), v.end());
	set<int>::iterator it = tmp.begin();
	it++;
	tmp.erase(it);
	set<int> ::iterator it1 = tmp.find(2);
	if (it1 != tmp.end()) 
	{cout << "找到了数据为:" << *it1 << endl;}
	else 
	{cout << "容器中没有这个数据" << endl;}
}

容器一开始装的数据为2,1,5,4,3,it一开始指向的是容器中的第一个元素 1,将it的值加加那么他就会指向容器中的第二个元素也就是2,当我们使用erase函数将迭代器指向的第二个元素删除之后再去查找数据2,便会发现找不到了
在这里插入图片描述
那么这就是erase函数的第一种形式,通过传递迭代器具体指向的位置来删除元素,这种形式的erase主要和find函数互相搭配使用,find函数找到元素之后会通过迭代器去的形式返回元素所在的位置,那么我们就可以通过erase的第一种形式将其删除,比如说下面的代码:

void func3()
{
	vector<int> v = { 2,1,5,4,3 };
	set<int> tmp(v.begin(), v.end());
	set<int>::iterator it = tmp.find(4);
	if (it != tmp.end())
	{
		tmp.erase(it);
	}
	it = tmp.find(4);
	if (it != tmp.end()) 
	{cout << "找到了数据为:" << *it << endl;}
	else 
	{cout << "容器中没有这个数据" << endl;}
}

这段代码的运行结果如下:
在这里插入图片描述
那么这就是erase函数的第一种形式的用法,第二种形式就是值删除比如说我想要删除容器种的3的话我就可以在erase函数种直接添加一个3,如果删除成功的话erase函数就会返回1如果删除的元素不存在导致删除失败的话就会返回0,比如说下面的代码:

void func3()
{
	vector<int> v = { 2,1,5,4,3 };
	set<int> tmp(v.begin(), v.end());
	cout << tmp.erase(3)<<endl;
	cout << tmp.erase(6) << endl;
}

这段代码的运行结果如下:
在这里插入图片描述

第三个就是迭代器区间的删除,这种删除方式就是将迭代器区间的内容给删除掉,当然这里的迭代器指的是set<T>的迭代器,比如说下面的代码:

void func3()
{
	vector<int> v = { 2,1,5,4,3,6,8,7 };
	set<int> tmp(v.begin(), v.end());
	tmp.erase(++tmp.begin(), --tmp.end());
}

这里的erase函数执行完之后容器中只会存在两个元素,一个是第一个元素,另外一个就是最后一个元素
在这里插入图片描述
那么这就是erase函数三种参数形式的使用方法以及特点希望大家能够理解。

count函数

我们来看看cout函数的参数和这个函数的作用:
在这里插入图片描述

cout函数的作用是返回这个值在树中出现的个数,这个函数的作用就是返回这个值在不在树里面,如果不在的话这个函数就会返回0,如果在的话这个函数就会这个数据在容器里面出现的个数,可能大家会感觉这个函数的存在属实有一点点鸡肋,因为find函数也可以干这种事情,那为什么还要这个函数呢?那这里就有两个原因,第一个就是conut函数可以更加方便的判断某个元素在不在,find函数的返回值是一个迭代器我们想要知道在不在的话得拿这个迭代器和容器的end函数做比较,而count函数则可以直接判断,第二个原因就是这个函数主要是为multiset进行服务的这个multiset也是可以看成set容器但是他允许数据冗余,也就是说容器中可以存在多个相同的数据,那么这就是count函数的作用希望大家能够理解。

lower_bound

upper_bound

我们首先来看看这两个函数的参数和介绍:
在这里插入图片描述
在这里插入图片描述

lower_bound和upper_bound的作用就是确定一个区间,lower_bound(x)找的就是大于等于x的位置upper_bound(y)找的就是大于y的位置,比如下面的代码:

void func4()
{
	vector<int> v = { 1,4,3,8,7,10,11,13 ,6};
	set<int> tmp(v.begin(), v.end());
	set<int>::iterator it_begin = tmp.lower_bound(5);
	cout << *it_begin << endl;
	set<int>::iterator it_end = tmp.upper_bound(8);
	cout << *it_end << endl;
}

因为容器中没有5所以lower_bound(5)找到的元素就是6,由于容器中存在8所以upper_bound(8)找的是大于8的元素的位置,所以upper_bound函数找到的就是10,比如说下面的代码:
在这里插入图片描述
这两个函数没什么用,可能唯一有用的地方就是搭配erase函数删除一段区间的元素给删除了,比如说下面的代码:

void func4()
{
	vector<int> v = { 1,4,3,8,7,10,11,13 ,6};
	set<int> tmp(v.begin(), v.end());
	set<int>::iterator it_begin = tmp.lower_bound(5);
	cout << *it_begin << endl;
	set<int>::iterator it_end = tmp.upper_bound(8);
	cout << *it_end << endl;
	tmp.erase(it_begin, it_end);
}

这里就会将大于等于6小于10的元素全部都给删除了,这里通过调试给大家看看erase函数的运行结果:
在这里插入图片描述
大家可以看到7和8都被删除了但是10没有被删除,那么这就是两个函数的作用希望大家可以理解。

multiset

multiset也是一个容器:
在这里插入图片描述

我们说set函数的作用是将一段数据进行排序和去重,当往set容器里面插入多个相同数据时,set容器只会存储一个数据,其他数据都会被过滤掉,但是multiset和set不一样,他不会对重复的数据进行过滤所以在这个容器里面就可以存在多个相同的数据,比如说下面的代码:

void func5()
{
	set<int> tmp1;
	tmp1.insert(1);
	tmp1.insert(1);
	tmp1.insert(1);
	multiset<int> tmp2;
	tmp2.insert(1);
	tmp2.insert(1);
	tmp2.insert(1);
}

通过调试便可以看到下面这两个容器的不同:
在这里插入图片描述
multiset容器和set容器的使用方法上没有什么区别,比如说conut函数这个函数在set中只能返回0或者1因为一个元素在set中也只能出现1次或者0次,但是在multiset中相同的元素会出现多次,所以count函数的返回值也会有多个,比如说下面的代码:

void func5()
{
	set<int> tmp1;
	tmp1.insert(1);
	tmp1.insert(1);
	tmp1.insert(1);
	cout << tmp1.count(1) << endl;
	multiset<int> tmp2;
	tmp2.insert(1);
	tmp2.insert(1);
	tmp2.insert(1);
	cout << tmp2.count(1) << endl;
}

这个代码的运行结果如下:
在这里插入图片描述
那么对于multiset大家可能会存在这么几个疑问,我们知道set的底层是通过搜索二叉树来实现的,对于搜索二叉树比根节点大的值会放到根节点的右边,比根节点小的值则会放到根节点的左边,multiset也是基于搜索二叉树实现的,那出现了相等的值这个值放到根节点的左边还是右边呢?答案是左边或者右边都可以,因为会发生旋转就算你放到左边也可能会因为旋转被放到右边,所以这个时候放到左边或者右边意义就没有那么大了所以multiset的insert函数就只是排序,对于multiset还有个问题就是容器中存在多个相同的数据,那find函数找的是容器中的哪个数据呢?答案是查找到的第一个数据就是find函数返回的,我们可以通过下面的代码来验证这一点:

void func6()
{
	vector<int> v = { 1,1,2,2,2,3,4,5,6 };
	multiset<int> tmp1(v.begin(),v.end());
	multiset<int>:: iterator it=tmp1.find(2);
	while (*it == 2)
	{
		cout << *it << endl;
		++it;
	}
}

这段代码的运行结果如下:
在这里插入图片描述
可以看到这里打印出来了三个2,所以这里的find找到就是第一个出现的数据,那么这就是multiset的用法即性质希望大家能够理解。

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

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

相关文章

DAY22:二叉树(十二)二叉搜索树最小绝对差+二叉搜索树中的众数

文章目录 530.二叉搜索树的最小绝对差思路完整版双指针优化写法&#xff1a;不用创建数组遍历pre root为什么是指向当前遍历的前一个节点 501.二叉搜索树中的众数&#xff08;这道题要知道普通二叉树怎么写&#xff09;思路完整版普通二叉树的写法sort自定义比较函数cmp的情况…

[论文阅读笔记77]LoRA:Low-Rank Adaptation of Large Language Models

1. 基本信息 题目论文作者与单位来源年份LoRA: Low-Rank Adaptation of Large Language ModelsmicrosoftInternational Conference on Learning Representations2021 524 Citations 论文链接&#xff1a;https://arxiv.org/pdf/2106.09685.pdf 论文代码&#xff1a;https://…

IMX6ULL裸机篇之SPI实验

一. SPI 实验 SPI实验&#xff1a;学习如何使用 I.MX6U 的 SPI 接口来驱动 ICM-20608&#xff0c;读取 ICM-20608 的六轴数据。 本文学习 SPI主控芯片的代码编写。其中&#xff0c;包括SPI工作模式设置&#xff0c;主从模式设置&#xff0c;时钟配置等实现。 二. SPI 主控芯…

EBU5476 Microprocessor System Design 知识点总结_2 Arm architecture

ARM架构 ARM是一个指令集&#xff0c;前面讲的几个汇编指令这些都算做指令。 ARM公司有意思的地方是&#xff0c;他们不做ARM设备&#xff0c;他们只设计指令集架构&#xff0c;然后授权&#xff08;知识产权核&#xff0c;IP核&#xff09;给其他半导体厂商做。 A&#xff…

大数据分析与机器学习:技术深度与实例解析【上进小菜猪大数据系列】

上进小菜猪&#xff0c;沈工大软件工程专业&#xff0c;爱好敲代码&#xff0c;持续输出干货。 大数据分析与机器学习已成为当今商业决策和科学研究中的关键组成部分。本文将深入探讨大数据技术的背景和原则&#xff0c;并结合实例介绍一些常见的大数据分析和机器学习技术。 …

chatgpt赋能python:Python如何拟合直线:使用最小二乘法

Python如何拟合直线&#xff1a;使用最小二乘法 在数据分析和机器学习领域&#xff0c;拟合一个最佳的直线是很常见和有用的。Python中有很多库可以拟合直线&#xff0c;但最小二乘法是其中最常用的一种方法。在本文中&#xff0c;我们将介绍最小二乘法的原理和如何在Python中…

【Goalng 开源项目】还在手写重复的 CRUD 吗?这个开源项目帮你解放双手

gormpher Gormpher 介绍快速开始WebObject 接口约定查询单条数据删除单条数据创建单条数据编辑单条数据条件查询多条数据 进阶WebObject 配置项动态接口函数Gorm 泛型函数Admin 源码handleEditObjecthandleQueryObject Gormpher 介绍 gormpher 是一个轻量级的 Golang 库 基于…

编译原理及应用期末复习

杂 3型文法 右线性文法 短语、直接短语、句柄 、判断是否是二义性文法 1、证明是二义性文法&#xff1a;证明存在一个句子有两颗不同的语法树 ① 画语法分析树 ② 、NFA、DFA K&#xff1a;所有状态&#xff0c;包含初始状态 Σ&#xff1a;终结字符集 M&#xff1a;状…

chatgpt赋能python:Python中替换字符串成int类型的方法

Python中替换字符串成int类型的方法 简介 在Python编程过程中&#xff0c;经常需要对字符串进行处理。有时候我们需要将字符串中的某些字符替换成int型数据&#xff0c;以便于进行一定的数值计算或其他操作。本文将介绍如何在Python中找出需要替换的字符&#xff0c;并将其转…

Python实现将txt文件转换成对应的excel或csv文件

前言 本文是该专栏的第29篇,后面会持续分享python的各种干货知识,值得关注。 工作上可能会遇到这样的情况,使用python将某个txt文本,按照行索引和列索引转换成对应的excel文件或者是csv文件。 那对于这样的需求,用python如何实现呢?跟着笔者直接往下看解决方法。(附完…

UART协议总结

UART&#xff08; Universal Asynchronous Receiver-Transmitter&#xff0c;通用异步收发器&#xff09;&#xff0c;是异步串行通信协议&#xff0c;用来传输串行数据。 1、UART原理说明 UART是全双工工作模式&#xff0c;其数据传输方法如下&#xff1a; 发送数据时&…

Hive 和 Oracle 中 Decode 函数的用法差异

前言 在数仓构建过程中,需要从业务那边进行数据的迁移!数仓大多数公司都是使用Hive,而业务那边使用的是Oracle数据库居多。最近就有个小伙伴在迁移的时候碰到了问题: 从报错来看,在使用 Decode() 函数的时候,传参有问题! 既然问题来了,我们就来解决一下呗,只有不断地…

基础知识学习---牛客网C++面试宝典(五)操作系统--第一节

1、本栏用来记录社招找工作过程中的内容&#xff0c;包括基础知识学习以及面试问题的记录等&#xff0c;以便于后续个人回顾学习&#xff1b; 暂时只有2023年3月份&#xff0c;第一次社招找工作的过程&#xff1b; 2、个人经历&#xff1a; 研究生期间课题是SLAM在无人机上的应…

LeetCode_Day6 | 四数相加||、赎金信、三数之和、四数之和!

LeetCode_哈希表 454.四数相加1.题目描述2.思路3.代码实现 383.赎金信1.题目描述2.暴力法3.哈希法思路代码实现 15.三数之和1.题目描述 454.四数相加 1.题目描述 详情leetcode链接 2.思路 解题步骤&#xff1a; 首先定义 map&#xff0c;key放a和b两数之和&#xff0c;valu…

神器CLIP:连接文本和图像,打造可迁移的视觉模型

2021年见证了vision transformer的大爆发&#xff0c;随着谷歌提出ViT之后&#xff0c;一大批的vision transformer的工作席卷计算机视觉任务。除了vision transformer&#xff0c;另外一个对计算机视觉影响比较大的工作就是Open AI在2021年1月份发布的DALL-E和CLIP&#xff0c…

chatgpt赋能python:一、Python在数据可视化中的应用

一、Python在数据可视化中的应用 Python是一种功能强大的编程语言&#xff0c;早已成为数据科学家和分析师的首选语言。数据可视化对于从数据中汲取信息和传达想法来说至关重要。Python也是数据可视化的理想工具之一。Python提供了许多强大的库&#xff0c;其中包括了一些流行…

( 2023版)互联网 Java 工程师面试题及答案汇总

最近很多粉丝朋友私信我说&#xff1a;熬过了去年的寒冬却没熬过现在的内卷&#xff1b;打开 Boss 直拒一排已读不回&#xff0c;回的基本都是外包&#xff0c;薪资还给的不高&#xff0c;对技术水平要求也远超从前&#xff1b;感觉 Java 一个初中级岗位有上千人同时竞争&#…

【野指针】

野指针 1. 指针是什么&#xff1f;2. 指针和指针类型2.1 指针-整数2.2 指针的解引用 3. 野指针3.1 野指针成因3.2 如何规避野指针 1. 指针是什么&#xff1f; 指针是什么&#xff1f; 指针理解的2个要点&#xff1a; 指针是内存中一个最小单元的编号&#xff0c;也就是地址平…

MySQL性能优化:慢查询优化

一、执行计划 执行计划的语法 在SQL查询的前面加上EXPLAIN关键字就行。比如: EXPLAIN select* from order_exp;执行效果如下。 &#xff08;一&#xff09;参数详解&#xff1a; 1、id 在一个大的查询语句中每个SELECT关键字都对应一个唯一的id。我们知道我们写的查询语句一…

【框架源码】Spring源码解析之BeanDefinition加载流程解析

观看本文之前&#xff0c;我们先思考一个问题&#xff0c;Spring是如何描述Bean对象的&#xff1f; Spring是根据BeanDefinition来创建Bean对象&#xff0c;BeanDefinition就是Spring中表示Bean定义。BeanDefinition用来存储Bean的相关信息&#xff0c;主要包括&#xff1a;Be…