【C++】map和set用法详解

news2024/11/14 5:23:01

文章目录

  • 1.关联式容器
  • 2.键值对
  • 3.树形结构的关联式容器
    • 3.1 set
      • 3.1.1 set的介绍
      • 3.1.2 set的模板参数列表
      • 3.1.3 set的使用
    • 3.2 map
      • map的介绍
      • map的模板参数列表
      • map的使用
      • 关于map的元素访问总结
    • 3.3multimap

1.关联式容器

我们接触过STL中的部分容器,比如:vector,list,deque,forward_list(C++11)等,这些容器统称为序列式容器,因为其底层为线性序列的数据结构,里面存储的是元素本身。那么什么是关联式容器?它与序列式容器有什么区别?

关联式容器也是用来存储数据的,与序列式容器不同的是,其里面存储的是<key,value>结构的键值对,在数据检索时比序列式容器效率更高。

2.键值对

用来表示具有一一对应关系的一种结构,该结构中一般只包含两个成员变量key和value,key代表键值,value表示与key对应的信息。比如:现在要建立一个英汉互译的字典,那该字典中必然有英文单词与其对应的中文含义,而且,英文单词与其中文含义是一一对应的关系,即通过该单词,在词典就可以找到与其对应的中文含义;比如每个学生的学号跟他的姓名,年龄,专业(类型可定义成数组)存在对应关系,找到学号,就可以查到这个学生的相关信息。

SGI-STL中关于键值对的定义:

//STL中关于键值对的定义
template<class T1,class T2>
struct pair
{
	typedef T1 first_type;
	typedef T2 first_type;

	T1 first;
	T2 second;

	//构造函数初始化
	pair()
		:first(T1())//key
		,second(T2())//value
	{}
	//拷贝构造函数,使用实例pair<string,int>
	pair(const T1& a,const T2& b)
		:first(a)
		,second(b)
	{}
};

3.树形结构的关联式容器

根据应用场景不同,STL总共实现了两种不同结构的关联式容器:树形结构与哈希结构。树形结构的关联式容器主要有四种:map,set,multimap,multiset。这四种容器的共同点是:使用平衡二叉树(即红黑树)作为底层,容器中的元素是一个有序的序列。下面依次介绍每一个容器。

3.1 set

3.1.1 set的介绍

1.set翻译为集合,是一个内部自动有序且不含重复元素的关联式容器。当我们想要去重,就可以利用到set,是一个很直观的接口,并且加入set之后可以实现自动排序,需要注意的是set的元素默认按照小于比较。
2、在set中,每个value必须是唯一的。set中的元素不能在容器中修改(元素总是const),但是可以在容器中插入或删除数据。一般的做法是先删除旧元素,然后添加新元素,这当然是为了维护里面元素的有序性。
3.set在底层使用二叉搜索树(红黑树)实现的。
4.与map/multimap不同,map/multimap中存储的是真正的键值对<key,value>,而set中只放value,但在底层实际存放的是由<value,value>构成的键值对。
5.在set中查找某个元素,时间复杂度为:log2N

3.1.2 set的模板参数列表

//头文件
#include<set>
std::set
template<class T,class Compare = less<T>,class Alloc = allocator<T>>

T:set中存放元素的类型,实际在底层存储<value, value>的键值对。
Compare(仿函数):set中元素默认按照小于来比较
Alloc:set中元素空间的管理方式,使用STL提供的空间配置器管理

3.1.3 set的使用

点此进入->set的文档

我在下列代码一一解释了关于set常见的函数该如何使用,

#include<iostream>
#include<set>
using namespace std;
int main()
{
	set<int> myset;//定义

	//迭代器的使用
	set<int>::iterator itlow, itup;

	for (int i = 1; i < 10; i++)
	{
		//插入数据
		//注意set会将元素自动排序
		myset.insert(i * 10);//10 20 30 40 50 60 70 80 90
	}

	//删除35-75之间的数据
	itlow = myset.lower_bound(35);//返回第一个大于等于key_value的定位器
	itup = myset.upper_bound(75);//返回最后一个小于等于key_value的定位器

	//注意:set中的删除操作不进行错误检查,所以用的时候最好用一下find()函数,
    //返回给定值定位器,如果没找到则返回end()。
	myset.erase(itlow, itup);

	/*for (auto it = myset.begin(); it != myset.end(); ++it)
	{
		cout << *it <<' ';
	}*/

	//这里可以使用范围for,需要加&,因为拷贝也是存在一定代价的,不需要修改的话,也尽量加const
	//范围for的底层是迭代器,操作由编译器完成
	for (auto& s : myset)
	{
		cout << s << ' ';
	}
	cout << endl;

	//count() 用来查找set中某个键值出现的次数。这个函数在set并不是很实用,在map中的用处比较大
	//因为一个键值在set只可能出现0或1次,==这样就变成了判断某一键值是否在set出现过==。
	cout << myset.count(100) << endl;//0不存在
	cout << myset.count(30) << endl;//1存在

	return 0;
}

3.2 map

map的介绍

点此进入->map的官方文档

1.map是关联式容器,它按照特定的次序(按照key来排序),存储由key和value组合而成的元素。
2.map中,是真正的键值对<key,value>,键值key和值value的类型可能不同,并且在map内部,key与value通过成员类型value_type绑定在一起,取名为pair。

typedef pair<const key,T> value_type
  1. 在内部,map中的元素总是按照键值key进行比较排序的。
  2. map支持下标访问,即在[]中放入key,就可以找到与key对应的value。在map中key不允许修改,key对应的value可以修改。
  3. map通常被实现为二叉搜索树(更准确的说:平衡二叉搜索树(红黑树))。

map的模板参数列表

相比set,map多了一个参数

#include<map>
std::map
template<class key,class T,class Compare = less<key>,class Alloc = allocator<pair<const Key,T>>

key: 键值对中key的类型
T: 键值对中value的类型
Compare: 比较器的类型,map中的元素是按照key来比较的,缺省情况下按照小于来比

Alloc:通过空间配置器来申请底层空间,不需要用户传递,除非用户不想使用标准库提供的
空间配置器
注意:在使用map时,需要包含头文件。

map的使用

map的官方文档已包括所有关于map的接口函数,现在我们来尝试使用它的常用接口。
最重要的应该有:插入函数insert,[ ]下标访问操作符,对于pair的理解,迭代器的使用,删除函数。

#include<iostream>
#include<map>
#include<string>
using namespace std;
int main()
{
	//定义一个map对象
	map<int, string> Stu;

	//1.用insert函数插入pair
	Stu.insert(pair<int, string>(01, "张三"));
	Stu.insert(pair<int, string>(02, "李四"));
	Stu.insert(pair<int, string>(03, "王五"));
	//但是我们发现在这里,写一个pair很不方便,因此我们可以用make_pair
	//make_pair的底层:是一个函数模板,还是调用pair去构造
	/*template<class T1,class T2>
	pair<T1,T2> make_pair(T1 x,T2 y)
	{
		return (pair<T1, T2>(x, y));
	}*/
	Stu.insert(make_pair(04, "李芳"));

	//3.第三种用[]的方式插入数据
	Stu[123] = "王五";

	//4.map的迭代器
	//map<int, string>::iterator it = Stu.begin();
	auto it = Stu.begin();

	while (it != Stu.end())
	{
		cout << it->first << ":" << it->second << endl;
		++it;
	}
	//最好加上引用,如果不修改的话,把const加上更好
	for (const auto& kv : Stu)
	{
		cout << kv.first << ":"<< kv.second << endl;
	}
	cout << endl;
	return 0;
}

活学活用:统计水果出现的次数

//利用map,统计水果出现的次数
	string arr[] = { "苹果", "西瓜", "香蕉", "草莓", "苹果", "西瓜", "苹果", "苹果", "西瓜", "苹果", "香蕉", "苹果", "香蕉" };
	map<string, int> countMap;
	for (auto& e: arr)
	{
		//map<string, int>::iterator it = countMap.find(e);
		auto it = countMap.find(e);//find函数找不到则返回end()
		if (it == countMap.end())
		{
			countMap.insert(make_pair(e,1));
		}
		else
		{
			it->second++;
		}
	}
	//有个更简洁的方法
	//for (auto& e : arr)
	//{
	//	countMap[e]++;
	//}

	for (const auto& e : countMap)
	{
		cout << e.first << ":" << e.second;
	}

关于map的元素访问总结

  1. map中的元素是键值对,我们需要学习pair,然后学会使用map容器,插入数据,管理数据。
  2. map中的key是唯一的,并且不能修改
  3. map默认按照小于(升序)的方式,并且是对key排序的。map中的元素如果用迭代器去遍历,是采用中序遍历的方式,可以得到一个有序序列。
  4. map的底层是一个平衡二叉树(红黑树),查找效率很高O(logN)
  5. map中[ ]下标访问操作符是一个大头,它支持查找,修改,插入。operator[ ]向map中插入元素的原理:用<key,T()>构造一个键值对,然后调用insert()函数将该键值对插入到map中,map中的键值对key一定是唯一的,如果key已经存在,插入失败,insert函数返回该key所在位置的迭代器。如果key不存在,插入成功,insert函数返回新插入元素所在位置的迭代器。
  6. operator[ ]函数最后将insert返回值键值对中的value返回。(也就是pair类中的second成员)
  7. 注意:在元素访问时,有一个与operat[ ]类似的函数,是at()函数,但是这个函数不常用,它们都是通过key找到与key对应的value然后返回其引用,不同的是:当key不存在时,operator[]会用默认的pair插入,返回该默认的value,而at()函数会直接抛异常。
  8. operator[]底层实现如下->
    在这里插入图片描述

3.3multimap

注意:multimap和map的唯一不同就是:map中的key是唯一的,而multimap中key是可以重复的。

multimap中的接口可以参考map,功能都是类似的。
注意:

  1. multimap中的key是可以重复的。
  2. multimap中的元素默认将key按照小于来比较。
  3. multimap中没有重载operator[]操作。
  4. 使用时与map包含的头文件相同。

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

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

相关文章

2 k-近邻算法

0 问题引入 想一想&#xff1a;下面图片中有三种豆&#xff0c;其中三颗豆品种未知&#xff0c;如何判断他们类型&#xff1f; 1 KNN概述 1.1 KNN场景 电影可以按照题材分类&#xff0c;那么如何区分 动作片 和 爱情片 呢&#xff1f; 动作片&#xff1a;打斗次数更多爱情…

【蓝牙mesh】Lower协议层介绍

【蓝牙mesh】Lower协议层介绍 Lower层简介 Lower协议层用于处理网络层以下的功能&#xff0c;包括节点的广播、重传、路由和网络拓扑等&#xff0c;是实现蓝牙mesh网络的关键协议之一。其中Lower协议层中最主要的一部分工作就是mesh数据的分片和组包。 Lower层是将Upper层发过…

buu [GWCTF 2019]BabyRSA 1

题目描述&#xff1a; import hashlib import sympy from Crypto.Util.number import *flag GWHT{******} secret ******assert(len(flag) 38)half len(flag) / 2flag1 flag[:half] flag2 flag[half:]secret_num getPrime(1024) * bytes_to_long(secret)p sympy.nextp…

【MySQL】事务隔离级别是怎么实现的?

事务隔离级别是怎么实现的&#xff1f; 四种隔离级别具体的实现方式 对于「读未提交」&#xff1a;直接读取最新的数据就好。对于「串行化」&#xff1a;通过加读写锁的方式来避免并行访问。对于「读提交」和「可重复读」&#xff1a;通过 Read View 来实现&#xff0c;主要区…

JS学习第4天——事件高级(注册/删除事件、DOM事件流、阻止事件冒泡、事件委托、常用的鼠标/键盘事件)

目录一、注册事件 / 删除事件1、注册事件&#xff08;绑定事件&#xff09;2、删除事件&#xff08;解绑事件&#xff09;二、DOM事件流三、事件对象event1、事件对象的兼容性2、e.targent和this的区别3、事件对象常见的属性和方法四、阻止事件冒泡五、事件委托&#xff08;代理…

数据采集与预处理学习

文章目录要求题解要求 根据表格求出哪两个演员合作电影数最多&#xff0c;及合作的电影数。 题解 from openpyxl import load_workbookwb load_workbook("电影导演演员信息表.xlsx") ws wb.worksheets[0]actors_films dict() for i, row in enumerate(ws.rows):i…

CAN总线开发一本全(4) - FlexCAN的驱动程序

CAN总线开发一本全&#xff08;4&#xff09; - FlexCAN的驱动程序 苏勇&#xff0c;2023年2月 文章目录CAN总线开发一本全&#xff08;4&#xff09; - FlexCAN的驱动程序引言从MindSDK获取FlexCAN驱动程序数据结构配置通信引擎的结构体类型访问MB的结构体类型配置ID过滤器的…

1 机器学习基础

1 机器学习概述 1.1 数据驱动的问题求解 大数据-Big Data 大数据的多面性 1.2 数据分析 机器学习&#xff1a;海量的数据&#xff0c;获取有用的信息 专门研究计算机怎样模拟或实现人类的学习行为&#xff0c;以获取新的知识或技能&#xff0c;重新组织已有的知识结构使之…

Python多进程编程

一 多进程编程 Python实现多进程的方式有两种&#xff1a;一种方法是os模块中的fork方法&#xff0c;另一种是使用multiprocessing模块。 前者仅适用于LINUX/UNIX操作系统&#xff0c;对Windows不支持&#xff0c;后者则是跨平台的实现方式。 第一种方式&#xff1a;使用os模…

【C++修行之路】STL——模拟实现string类

文章目录前言类框架构造与析构c_str迭代器操作符重载[]&#xff1a;&#xff1a;> > < < !:reverse与resizereverseresizepush_back与append复用实现insert和erasec_str与流插入、流提取eraseswap(s1,s2)与s1.swap(s2)结语前言 这次我们分几个部分来实现string类…

spark第一章:环境安装

系列文章目录 spark第一章&#xff1a;环境安装 文章目录系列文章目录前言一、文件准备1.文件上传2.文件解压3.修改配置4.启动环境二、历史服务器1.修改配置2.启动历史服务器总结前言 spark在大数据环境的重要程度就不必细说了&#xff0c;直接开始吧。 一、文件准备 1.文件…

React Use Hook 尝鲜

React Use Hook 尝鲜 最近继续在找处理 React 异步调用的方式……主要是现在需求比较复杂&#xff0c;用 cache query 的方式去实现有那么一丢丢的麻烦&#xff0c;又不是很想用额外的包&#xff0c;所以就想看看有没有比较好的一些处理方式。 当然&#xff0c;可以用到生产环…

tkinter界面的TCP通信/tkinter开启线程接收TCP

前言 用简洁的语言写一个可以与TCP客户端实时通信的界面。之前做了一个项目是要与PLC进行信息交互的界面&#xff0c;在测试的时候就利用TCP客户端来实验&#xff0c;文末会附上TCP客户端。本文分为三部分&#xff0c;第一部分是在界面向TCP发送数据&#xff0c;第二部分是接收…

Linux基础命令-dd拷贝、转换文件

文章目录 dd 命令介绍 语法格式 基本参数 参考实例 1&#xff09;生成一个200M的新文件 2&#xff09;拷贝文件的100个字节 3&#xff09;将文件的字母全部转换成大写 4&#xff09;将linux自带的光盘制作成iso格式的镜像文件 5&#xff09;使用dd命令制作1G的交换分…

软考中级-操作系统

1 操作系统地位计算机系统由硬件和软件组成&#xff0c;未配置软件的称为裸机&#xff0c;但这会导致效率低下。操作系统是为弥补用户与硬件之间的鸿沟的一种系统软件&#xff0c;汇编、编译、解释、数据库管理系统等系统软件和其他应用软件都在此基础。2 进程管理又称处理机管…

Linux Ubuntu配置国内源

因为众所周知的原因&#xff0c;国外的很多网站在国内是访问不了或者访问极慢的&#xff0c;这其中就包括了Ubuntu的官方源。 所以&#xff0c;想要流畅的使用apt安装应用&#xff0c;就需要配置国内源的镜像。 市面上Ubuntu的国内镜像源非常多&#xff0c;比较有代表性的有清华…

pytorch学习日记之激活函数

常用的激活函数为S型&#xff08;sigmoid&#xff09;激活函数、双曲正切&#xff08;Tanh&#xff09;激活函数、线性修正单元&#xff08;ReLU&#xff09;激活函数等&#xff0c;对应Pytorch的函数如下所示 层对应的种类功能torch.nn.SigmoidSigmoid激活函数torch.nn.TanhT…

_vue-3

Vue3有了解过吗&#xff1f;能说说跟vue2的区别吗&#xff1f; 1. 哪些变化 从上图中&#xff0c;我们可以概览Vue3的新特性&#xff0c;如下&#xff1a; 速度更快体积减少更易维护更接近原生更易使用 1.1 速度更快 vue3相比vue2 重写了虚拟Dom实现编译模板的优化更高效的…

数据挖掘概述

目录1、数据挖掘概述2、数据挖掘常用库3、模型介绍3.1 分类3.2 聚类3.3 回归3.4 关联3.5 模型集成4、模型评估ROC 曲线5、模型应用1、数据挖掘概述 数据挖掘&#xff1a;寻找数据中隐含的知识并用于产生商业价值 数据挖掘产生原因&#xff1a;海量数据、维度众多、问题复杂 数…

直接拿项目运行npm start 会出现’react-scripts’ 不是内部或外部命令,也不是可运行的程序或批处理文件错误

目录 解决方案 原因 解决方案 npm install react-scripts或npm install安装完成后再次运行 npm start 即可 原因 create-react-app有丢包的缺陷&#xff0c;手动安装包后&#xff0c;需要重新npm install一下&#xff0c;这样node_modules/.bin/目录下才会重新出现react-s…