C++笔记16•数据结构:关联式容器map和set•

news2025/4/5 9:01:15

map和set

1.关联式容器

前面介绍的的是序列式容器:vectorlist、deque等容器。这次博客介绍STL新的容器成员,那就是关联式容器;顾名思义关联式容器就是容器存在中的数据之间存在联系(关联)。与序列式容器不同的是,其里面存储的是<key, value>结构的键值对,在数据检索时比序列式容器效率更高。一般key不能被修改,由key去查找key对应的value。

2. 键值对 

2.1含义:
用来表示具有一一对应关系的一种结构,该结构中一般只包含两个成员变量 key value key 表键值, value 表示与 key 对应的信息
2.2举例:
比如:现在要建立一个英汉互译的字典,那该字典中必然有英文单词与其对应的中文含义,而且,英文单词与其中文含义是一一对应的关系,即通过该应该单词,在词典中就可以找到与其对应的中文含义。
2.3键值对的结构
键值对是被存放在一个pair的结构体当中的
template <class T1, class T2>
struct pair
{
typedef T1 first_type;
typedef T2 second_type;
T1 first;
T2 second;
pair(): first(T1()), second(T2())
{}
pair(const T1& a, const T2& b): first(a), second(b)
{}
};

pair传参需要写参数类型,为了简便,pair被写在一个模板里叫做make_pair中省略了参数类型。

template <class T1,class T2>
  pair<T1,T2> make_pair (T1 x, T2 y)
  {
    return ( pair<T1,T2>(x,y) );
  }

举个例子:

#include <iostream>     

int main() {
	std::pair <int, int> foo;
	std::pair <double, int> bar;

	foo = std::make_pair(10, 20);
	bar = std::make_pair(10.5, 'A'); 

	std::cout << "foo: " << foo.first << ", " << foo.second << '\n';
	std::cout << "bar: " << bar.first << ", " << bar.second << '\n';

	return 0;
}

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

        STL 总共实现了两种不同结构的管理式容器:树型结构与哈希结构(后面介绍)
树型结 构的关联式容器主要有四种: map set multimap multiset 。这四种容器的共同点是:使用平衡搜索树( 即红黑树 ) 作为其底层结果,容器中的元素是一个有序的序列。下面一依次介绍每一个容器。

 ​​​3.1 set

   1. set 是按照一定次序存储元素的容器
   2. set 中,元素的 value 也标识它 (value 就是 key ,类型为 T) ,并且每个 value 必须是唯一的,set中的元素不能在容器中修改 ( 元素总是 const) ,但是可以从容器中插入或删除它们。
   3. 在内部, set 中的元素总是按照其内部比较对象 ( 类型比较 ) 所指示的特定严格弱排序准则进行排序。
   4. set 容器通过 key 访问单个元素并且 允许根据顺序对子集进行直接迭代。
   5. set 在底层是用二叉搜索树 (红黑树) 实现的。
   6.set 中插入元素时,只需要插入 value 即可,不需要构造键值对
   7.set 中的元素不可以重复 ( 因此可以使用 set 进行去重 )
   8.使用 set 的迭代器遍历 set 中的元素,可以得到有序序列(升序
   9.set中的元素默认按照小于来比较
  10.set 中查找某个元素,时间复杂度为:O( log(n))
代码示例:
int main()
{
	set<int> s;
	s.insert(1);
	s.insert(2);
	s.insert(3);
	s.insert(4);
	s.insert(5);
	s.insert(0);
	s.insert(6);
	s.insert(5);//set 不会将重复的5插入(去重)

	set<int>::iterator it = s.begin();
	while (it != s.end())
	{
		cout << *it << " ";
		++it;
	}
	cout <<endl;

	return 0;
}

3.2 multiset

与set的区别就是multiset可以插入重复数据,其他基本都一样,所包含的头文件都是 #include <set>
代码示例:
int main()
{
	multiset<int> s;

	//插入操作
	s.insert(1);
	s.insert(2);
	s.insert(3);
	s.insert(4);
	s.insert(5);
	s.insert(0);
	s.insert(6);
	s.insert(5);//multiset 会将重复的5插入  这个是与set最主要的区别

	multiset<int>::iterator it = s.begin();
	while (it != s.end())
	{
		cout << *it << " ";
		++it;
	}
	cout << endl;


	//查找操作
	multiset<int>::iterator pos = s.find(6);//时间复杂度O(logN)
	if (pos != s.end())
	{
		cout << "找到6了" << endl;
	}
	else
	{
		cout << "没有找到6" << endl;
	}

	pos = find(s.begin(), s.end(), 0);//时间复杂度O(N)
	if (pos != s.end())
	{
		cout << "找到0了" << endl;
	}
	else
	{
		cout << "没有找到0" << endl;
	}

	//重复的数据,find返回中序第一个数据
	multiset<int>::iterator pf = s.find(5);
	while (pf != s.end())
	{
		cout << *pf << " ";
		++pf;
	}
	cout << endl;

	//删除操作
	s.erase(0);//指定数值删除

	multiset<int>::iterator pose = s.find(10);//先查找再删除
	if (pose != s.end())
	{
		s.erase(pose);//删除迭代器
	}
	else
	{
		cout << "不存在,删除失败!" << endl;
	}
	multiset<int>::iterator start = s.lower_bound(2);  
	multiset<int>::iterator end = s.upper_bound(4);
	s.erase(start,end);//删除区间迭代器(删除给定数值区间的数据)

	for (auto i : s)
	{
		cout << i << " ";
	}
	cout << endl;

	return 0;

}

3.3 map

  1. map 是关联容器,它按照特定的次序 ( 按照 key 来比较 ) 存储由键值 key 和值 value 组合而成的元素。
  2. map 中,键值 key 通常用于排序和惟一地标识元素,而值 value 中存储与此键值 key 关联的内容。键值key 和值 value 的类型可能不同,并且在 map 的内部, key value 通过成员类型
value_type 绑定在一起,为其取别名称为 pair: typedef pair<const key, T> value_type;
  3. 在内部, map 中的元素总是按照键值 key 进行比较排序的。
  4. map 中通过键值访问单个元素, map 允许根据顺序对元素进行直接迭代( 即对 map 中的元素进行迭代时,可以得到一个有序的序列(默认升序 )
  5. map 支持下标访问符,即在 [] 中放入 key ,就可以找到与 key 对应的 value
  6. map 通常被实现为二叉搜索树 ( 更准确的说:平衡二叉搜索树 (map的底层是 红黑树 ))
  7.map中的的元素是键值对,封装在pair结构体中用first和second访问,key 是唯一的,并且不能修改,为了简便,pair被写在一个模板里叫做make_pair中省略了参数类型。
  8. map的底层为平衡搜索树 ( 红黑树 ) ,查找效率比较高 O(log(N))
  9. 支持 [ ] 操作符, operator[ ] 中实际进行插入查找。
说明: map中的[ ]操作符可以直接实现插入,其底层是用insert实现的;
代码示例:
#include <iostream>     
#include <map>
using namespace std;


void test1()
{
	map<int, int> m;
	m.insert(pair<int, int>(1, 1));
	m.insert(make_pair(2, 1));
	m.insert(make_pair(3, 1));
	m.insert(make_pair(4, 1));//用模板make_pair代替pair<int,int>
	//operator[] 实现插入
	m[5];//默认的value是0
	//m[5];//第二次插入会失败 因为map不支持插入重复的数据
	m[5]++;//第二次插入虽然会失败 但是m[5](operator[] 操作符)会返回原来节点的迭代器,并返回value值 由于m[5]还有后置++ 则value++
	m[6]++;//插入成功((operator[] 操作符)会返回新节点的迭代器) 并且value++
	m[6] = 3;//有了就只修改  (修改)
	m[7] = 2;//没有就 先插入,再修改(插入+修改)
	//m.insert({ 8, 1 });//直接插入


	map<int, int>::iterator it = m.begin();
	while (it != m.end())
	{
		cout << (*it).first << ":" << (*it).second << endl;
		cout << it->first << ":" << it->second << endl;
		++it;//一定要++it,更新迭代器
	}
	for (auto e : m)
	{
		cout << e.first << ":" << e.second << endl;

	}

}
void test2()
{
	string arr[] = { "葡萄", "梨", "哈密瓜", "西瓜", "苹果", "橙子", "苹果", "苹果", "西瓜", "苹果", "香蕉", "苹果", "香蕉" };
	map<string, int> m;


	//统计arr中水果的个数

	方法一 查找+插入
	//for(size_t i=0;i<sizeof(arr)/ sizeof(arr[0]);i++)
	//{
	//	map<string, int>::iterator pos = m.find(arr[i]);
	//	if (pos != m.end())
	//	{
	//		pos->second++;
	//	}
	//	else
	//	{
	//		m.insert(make_pair(arr[i], 1));
	//	}
	//}
	//范围for实现第二种的话
	for (auto& e : arr)
	{
		map<string, int>::iterator pos = m.find(e);
		if (pos != m.end())
		{
			pos->second++;
		}
		else
		{
			m.insert(make_pair(e, 1));
		}
	}
	
	//方法二
	//for (auto& s : arr)
	//{
	//	pair<map<string, int>::iterator,bool> ret = m.insert(make_pair(s,1));//简单写法 auto ret = m.insert(make_pair(s,1));初学者最好不要这样使用
	//	//auto ret = m.insert(make_pair(s, 1));
	//	if (!ret.second)//if (ret.second==false)
	//	{
	//		ret.first->second++;
	//	}
	//}

	方法三
	for (auto& e : arr)
	{
		m[e]++;
	}

	这三种统计次数的方法 一般推荐使用 方法三 使用operator[] 运算符重载


	map<string, int>::iterator it = m.begin();
	while (it != m.end())
	{
		cout << it->first << ":" << it->second << endl;
		++it;//一定要++it,更新迭代器
	}
}

int main()
{

	//test1();
	test2();
	return 0;
}

3.4multimap

        multimap与map的区别就是multimap可以插入重复数据,其他基本都一样,所包含的头文件都是 #include <map>
multimap 中的接口可以参考 map ,功能都是类似的。
注意:
1. multimap 中的 key 是可以重复的。
2. multimap 中的元素默认将 key 按照小于来比较
3. multimap 中没有重载 operator[] 操作 ( 同学们可思考下为什么 ?)
4. 使用时与 map 包含的头文件相同
ps: multimap map 的唯一不同就是: map 中的 key 是唯一的,而 multimap key 是可以
重复的。还有就是multimap map成员函数中 insert 返回值不一样:
代码示例
void test3()//multimap
{
	multimap<int, int> m;
	m.insert(pair<int, int>(1, 1));
	m.insert(make_pair(2, 1));
	m.insert(make_pair(3, 1));
	m.insert(make_pair(4, 1));//用模板make_pair代替pair<int,int>

	//multimap不支持operator[] 实现插入 因为多个相同的key 不知道指向哪一个value进行访问  但可以用原生的insert插入重复的数据
	m.insert(make_pair(5, 1));
	m.insert(make_pair(5, 1));
	m.insert({6,1});


	multimap<int, int>::iterator it = m.begin();
	while (it != m.end())
	{
		cout << it->first << " " << it->second << endl;
		++it;//一定要++it,更新迭代器
	}
	for (auto e : m)
	{
		cout << e.first << " " << e.second << endl;

	}
}

void test4()//multimap
{
	string arr[] = { "葡萄", "梨", "哈密瓜", "西瓜", "苹果", "橙子", "苹果", "苹果", "西瓜", "苹果", "香蕉", "苹果", "香蕉" };
	
	multimap<string, int> m;


	//统计arr中水果的个数

	方法一 查找+插入
	//for(size_t i=0;i<sizeof(arr)/ sizeof(arr[0]);i++)
	//{
	//	multimap<string, int>::iterator pos = m.find(arr[i]);
	//	if (pos != m.end())
	//	{
	//		pos->second++;
	//	}
	//	else
	//	{
	//		m.insert(make_pair(arr[i], 1));
	//	}
	//}
	范围for实现第二种的话
	for (auto& e : arr)
	{
		multimap<string, int>::iterator pos = m.find(e);
		if (pos != m.end())
		{
			pos->second++;
		}
		else
		{
			m.insert(make_pair(e, 1));
		}
	}

	//方法二
	//for (auto& s : arr)
	//{
	//	pair<multimap<string, int>::iterator, bool> ret = m.insert(make_pair(s, 1));//简单写法 auto ret = m.insert(make_pair(s,1));初学者最好不要这样使用
	//	//auto ret = m.insert(make_pair(s, 1));
	//	if (!ret.second)//if (ret.second==false)
	//	{
	//		ret.first->second++;
	//	}
	//}//方法二也不行 这个是因为multimap和map成员函数中的insert返回值不一样 
	// //map中的insert返回pair<iterator, bool>    multimap中的insert只返回iterator 

	//方法三
	//for (auto& e : arr)
	//{
	//	m[e]++;
	//}  //multimap不支持operator[]重载 此方法不可以使用



	multimap<string, int>::iterator it = m.begin();
	while (it != m.end())
	{
		cout << it->first << ":" << it->second << endl;
		++it;//一定要++it,更新迭代器
	}

	//实现全部插入(重复的也可以)不去重
	for (auto& e : arr)
	{
		m.insert(make_pair(e, 1));
	}
	for (auto& i : m)
	{
		cout << i.first << ":" << i.second << endl;
	}
}


int main()
{
	test3();
	test4();
	return 0;
}

 

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

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

相关文章

linux入门系列【1】常用命令

一、简介 linux 基本操作命令,便于我们去使用命令帮助我们去检索和排查问题 二、常用命令 1.磁盘空间排查 1.1 查看磁盘空间分布情况 du -ah .|sort -hr 在对应目录下执行则是查看对应目录的文件分布以及大小情况,一般用于查看某个文件夹目录数据情况 1.2 查看深度层级为…

Java中的Set(如果想知道Java中有关Set的知识点,那么只看这一篇就足够了!)

前言&#xff1a;在Java编程中&#xff0c;集合框架&#xff08;Collections Framework&#xff09;是处理数据结构和算法的基础工具之一。它提供了一套强大且灵活的接口和类&#xff0c;用于存储和操作不同类型的数据集合。在这其中&#xff0c;Set接口扮演着一个重要角色。与…

vue-router基本流程及其案例分析

web发展历程 1.后端实现路由 在这个阶段&#xff0c;前端基本上只写界面&#xff0c;也就是html,css,js那些东西&#xff0c;然后在界面中挖槽用来接后端数据&#xff0c;包括路由也由后端负责&#xff0c;在这个阶段中&#xff0c;web开发非常依赖后端&#xff0c;常见的后端…

系列精选 |【梧桐数据库】产品架构层次解析-总述

梧桐数据库中秋特别活动免费领取大闸蟹 抽奖免费领取大闸蟹 以下是正文 在浩瀚的数据世界里&#xff0c;梧桐数据库犹如一颗璀璨的星辰&#xff0c;它的设计如同一首细腻的诗歌&#xff0c;每一个层次都是优美的韵律&#xff0c;为我们构建了一个强大而灵动的数据天地。 梧桐数…

西中区2024年度安全知识竞赛活动方案

为有效预防安全生产事故的发生&#xff0c;深化西中区全体员工对安全生产的认识&#xff0c;切实提升全体人员的安全意识和自我保护能力&#xff0c;夯实安全知识基础&#xff0c;丰富安全文化内涵&#xff0c;推动安全生产工作更加规范化、系统化&#xff0c;根据西中区安全生…

<数据集>遥感航拍飞机和船舶和识别数据集<目标检测>

数据集格式&#xff1a;VOCYOLO格式 图片数量&#xff1a;19973张 标注数量(xml文件个数)&#xff1a;19973 标注数量(txt文件个数)&#xff1a;19973 标注类别数&#xff1a;2 标注类别名称&#xff1a;[ship,plane] 序号类别名称图片数框数1ship17575416292plane239815…

简单好用的SD卡克隆软件:轻松克隆SD卡

想更换SD卡以提升性能&#xff0c;但不知道如何进行SD卡克隆&#xff1f;不用担心&#xff0c;本文推荐了一款好用SD卡克隆软件&#xff0c;轻松帮你解决问题&#xff01; 为什么要克隆SD卡&#xff1f; SD卡广泛应用于游戏机、手机及其他便携设备。用户常用SD卡存储个人数据…

2024/9/3黑马头条跟学笔记(一)

D1 视频链接 Day1-05-nacos环境搭建_哔哩哔哩_bilibili 内容介绍 搭建微服务开发环境&#xff0c;登录接口包含注册中心和nacos配置中心 服务端用户…微服务。网关负载均衡转发接口请求 实现微服务间互相通信 接口测试 前后端联调 前置知识 背景介绍 类似今日头条&#x…

权威解读:社交类APP都需要办理哪些资质?

今天小编给大家讲讲社交类APP都需要办理哪些资质&#xff1f; 我们先来看下微信小程序对社交类目是怎么分类以及需要哪些资质许可证&#xff1f; 微信小程序社交类目许可资质 微信小程序对社交类目做了一些细分&#xff0c;它把社交分为陌生人交友、熟人交友、社区/论坛、直播…

log4j 控制台和文件输出乱码问题解决

一个小问题&#xff0c;却让我感觉到&#xff0c;现在真正动脑的人很少。。我来说说吧。 今天遇到一个小问题&#xff0c; log4j输出到文件乱码&#xff0c;控制台正常。显然是编码问题导致。Google一搜&#xff0c;几乎一水的说&#xff1a; 项目中log4j在英文版linux下输出中…

气膜水产养殖:打造高效、可持续的水产养殖新模式—轻空间

随着全球对高质量水产品需求的不断增加&#xff0c;传统的水产养殖方式面临着诸多挑战&#xff0c;如环境污染、气候变化以及水源短缺等问题。在这种背景下&#xff0c;气膜水产养殖作为一种创新的养殖模式&#xff0c;逐渐引起了广泛关注。通过结合气膜结构建筑与现代化养殖技…

【测试】系统测试用例编写案例模板(Word原件)

1编写目的 2使用范围 3文档概述 4术语和缩略语 5编写规范 5.1编写目的 5.2编写范围 5.3编写规范 6参考文档 软件全套精华资料包清单部分文件列表&#xff1a; 工作安排任务书&#xff0c;可行性分析报告&#xff0c;立项申请审批表&#xff0c;产品需求规格说明书&#xff0c;需…

从UGC到PGC:3C品牌与TikTok达人合作的内容优化策略

在数字营销新时代&#xff0c;内容的创作和传播方式正在经历快速的变革。3C品牌与TikTok达人的合作正逐渐从用户生成内容&#xff08;UGC&#xff09;向专业生成内容&#xff08;PGC&#xff09;转变。这一转变不仅改变了内容的生产方式&#xff0c;也提升了品牌营销的效果。本…

三种权限模型该如何选择

在构建企业级平台或复杂应用系统时&#xff0c;权限管理是一个至关重要的环节。它决定了哪些用户可以访问哪些资源&#xff0c;以及可以进行哪些操作&#xff0c;一个健全的权限管理架构&#xff0c;在确保系统正常运行的同时&#xff0c;也能有效防止数据泄露和非法访问&#…

antd:手写走马灯vue组件

在使用ant-design-vue做走马灯的时候,封装的组件的自由度太低,难以实现想要的效果,于是本人自己写了一个走马灯组件,以方便代码复用。本文将介绍如何在vue框架中,使用ant-design-vue手动实现走马灯组件效果。 结果如下图所示, 一、使用说明 使用时,直接创建一个组件,…

.NET 最好用的验证组件 FluentValidation

目录 前言 项目介绍 项目使用 1、安装FluentValidation 2、Program.cs 3、Startup.cs 4、版本兼容 5、支持的验证器 6、可扩展 7、Swagger 模型和验证器 8、包含验证器 高级用法 1、异步验证 2、条件验证 3、自定义验证规则 4、自定义错误消息 项目地址 总结 …

comfyui替换电商模特工作流,模特们要真的要失业了吗?

前言 comfyui生态的丰富绝对是电商行业的福利&#xff0c;有助于电商老板们开源节流&#xff0c;废话不多说本着追求进步进一步理解comfyui工作流的搭建逻辑&#xff0c;我们来拆解电商模特替换这个工作流&#xff01; 老规矩一句话说工作流原理&#xff0c;1.借助XL-tile修改…

Funsound: 快速为你的视频加上字幕

Funsound是基于阿里达摩院funasr开发的中文语音识别工具&#xff0c;其paraformer非自回归解码速度超快&#xff0c;同时预训练模型识别精度业界领先。本文将简要介绍funsound下如何快速为你的视频添加字幕&#xff0c;十分简单方便。 1. 上传音视频识别 & 导出SRT 打开fu…

无人机之飞行速度篇

无人机的飞行速度是一个复杂且多变的参数&#xff0c;它受到多种因素的影响。以下是对无人机飞行速度及其影响因素的详细分析&#xff1a; 一、无人机飞行速度概述 无人机的飞行速度通常以其在不同飞行模式下的水平飞行速度来衡量&#xff0c;如平稳挡&#xff08;Cine&#x…

关于武汉高芯coin417G2红外机芯的二次开发

文章目录 前言一、外观和机芯参数二、SDK的使用1、打开相机2、回调函数中获取全局温度和图像3、关闭相机 前言 最近工作中接触了一款基于武汉高芯科技有限公司开发的红外模组,即coin417g2(测温型)9.1mm镜头.使用此模组,开发了一套红外热成像检测桌面应用程序.下面简单记录下该…