map/set和unordered_map/unordered_set的区别及其适用情况

本专栏内容为:C++学习专栏,分为初阶和进阶两部分。 通过本专栏的深入学习,你可以了解并掌握C++。
💓博主csdn个人主页:小小unicorn

⏩专栏分类:C++

🚚代码仓库:小小unicorn的代码仓库🚚

🌹🌹🌹关注我带你学习编程知识

目录

map/set与unordered_map/unordered_set的区别

xxx容器与unordered_xxx容器对上层所提供函数接口的名字和功能虽然一致,但它们的底层实现却大不相同,xxx容器和unordered_xxx容器的区别如下:

容器 底层数据结构 是否有序 实现版本 增删查改的效率 迭代器类型
unordered_map/unordered_set 哈希表/散列表 遍历无序 C++11 O ( 1 ) 单向迭代器
map/set 红黑树 遍历有序 c++98 O(logN) 双向迭代器

map/set与unordered_map/unordered_set的性能测试

map容器与unordered_map容器的差别和set容器与unordered_set容器的差别类似,下面我们以set容器和unordered_set容器的测试为例。

说到一个容器的性能,我们最关心的实际就是该容器增删查改的效率。我们可以通过下列代码测试set容器和unordered_set容器insert、find以及erase的效率。

cpp 复制代码
#include <iostream>
#include <set>
#include <unordered_set>
#include <time.h>
using namespace std;
int main()
{
	int N = 1000;
	vector<int> v;
	v.reserve(N);
	srand((unsigned int)time(NULL));
	//随机生成N个数字
	for (int i = 0; i < N; i++)
	{
		v.push_back(rand());
	}
	/****************插入效率测试****************/
	//将这N个数插入set容器
	set<int> s;
	clock_t begin1 = clock();
	for (auto e : v)
	{
		s.insert(e);
	}
	clock_t end1 = clock();
	//将这N个数插入unordered_set容器
	unordered_set<int> us;
	clock_t begin2 = clock();
	for (auto e : v)
	{
		us.insert(e);
	}
	clock_t end2 = clock();
	//分别输出插入set容器和unordered_set容器所用的时间
	cout << "set insert: " << end1 - begin1 << endl;
	cout << "unordered_set insert: " << end2 - begin2 << endl;
	/****************查找效率测试****************/
	//在set容器中查找这N个数
	clock_t begin3 = clock();
	for (auto e : v)
	{
		s.find(e);
	}
	clock_t end3 = clock();
	//在unordered_set容器中查找这N个数
	clock_t begin4 = clock();
	for (auto e : v)
	{
		us.find(e);
	}
	clock_t end4 = clock();
	//分别输出在set容器和unordered_set容器中查找这N个数所用的时间
	cout << "set find: " << end3 - begin3 << endl;
	cout << "unordered_set find: " << end4 - begin4 << endl;
	/****************删除效率测试****************/
	//将这N个数从set容器中删除
	clock_t begin5 = clock();
	for (auto e : v)
	{
		s.erase(e);
	}
	clock_t end5 = clock();

	//将这N个数从unordered_set容器中删除
	clock_t begin6 = clock();
	for (auto e : v)
	{
		us.erase(e);
	}
	clock_t end6 = clock();
	//分别输出将这N个数从set容器和unordered_set容器中删除所用的时间
	cout << "set erase: " << end5 - begin5 << endl;
	cout << "unordered_set erase: " << end6 - begin6 << endl;
	return 0;
}

对1000个数做增删查改操作

当N为1000时,set容器和unordered_set容器增删查改的效率差异并不大,在Debug版本下的测试结果如下:

在Release版本下,set容器和unordered_set容器对1000个数做增删查改操作所用的时间更是被优化到了接近0毫秒。

对10000000个数做增删查改操作

而当N为10000000时,set容器和unordered_set容器增删查改的效率的差异就很明显了,在Debug版本下的测试结果如下:

而经过Release版本优化后,unordered_set容器对10000000个数做查找和删除操作所用的时间还是被优化到了接近0毫秒,插入操作所用的时间也仅仅是0.1秒左右。但是set容器对10000000个数做增删查改操作所用的时间还是需要1秒左右,与unordered_set容器相比就占了下风。

注意: 这里给出的N值仅作参考,在不同的测试环境下可能不同。

结论

根据测试结果可以得出以下结论:

  1. 当处理数据量小时,map/set容器与unordered_map/unordered_set容器增删查改的效率差异不大。
  2. 当处理数据量大时,map/set容器与unordered_map/unordered_set容器增删查改的效率相比,unordered系列容器的效率更高。
  3. 因此,当处理数据量较小时,选用xxx容器与unordered_xxx容器的差异不大;当处理数据量较大时,建议选用对应的unordered_xxx容器。

补充一点: 当需要存储的序列为有序时,应该选用map/set容器。

相关推荐
红石程序员1 小时前
VSCode配置C++项目全攻略
开发语言·c++·visual studio
liulilittle2 小时前
通过高级处理器硬件指令集AES-NI实现AES-256-CFB算法并通过OPENSSL加密验证算法正确性。
linux·服务器·c++·算法·安全·加密·openssl
十秒耿直拆包选手2 小时前
Qt:Qt桌面程序正常退出注意事项
c++·qt
范纹杉想快点毕业4 小时前
初探Qt信号与槽机制
java·c语言·开发语言·c++·qt·visualstudio·visual studio
君鼎4 小时前
C++操作系统与网络编程(针对特定岗位)
网络·c++
一条叫做nemo的鱼5 小时前
从汇编的角度揭开C++ this指针的神秘面纱(上)
汇编·c++·算法·函数调用·this指针·参数传递
C++ 老炮儿的技术栈5 小时前
visual studio 2022更改主题为深色
c语言·开发语言·c++·ide·windows·git·visual studio
白总Server7 小时前
GaussDB 分布式数据库调优(架构到全链路优化)
java·网络·c++·架构·go·scala·数据库架构
whoarethenext7 小时前
C++/OpenCV地砖识别系统结合 Libevent 实现网络化 AI 接入
c++·人工智能·opencv
Antonio9157 小时前
【Linux】Linux基础I/O
linux·c++