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

map/set和unordered_map/unordered_set的区别

容器底层数据结构是否有序实现版本复杂度迭代器
map/set红黑树有序C++98O(logN)双向迭代器
unordered_map/unordered_set哈希表/散列表无序C++11O(1)单向迭代器

unordered_set无序的(VS下)

void unordered_set1()
{
	unordered_set<int> s = { 1,2,3,84965,165,1,651,63,6,23 };
	unordered_set<int>::iterator it = s.begin();
	while (it != s.end())
	{
		cout << *it << " "; //1 2 3 651 165 84965 63 6 23
		it++;
	}
	cout << endl;
}

 unordered_map(VS下)

void unordered_map1()
{
	string arr[] = { "苹果","西瓜","草莓","苹果","西瓜","草莓" ,"苹果","西瓜","草莓" ,"香蕉" };
	unordered_map<string,int> countMap;
	for (auto& e : arr)
	{
		countMap[e]++;
	}
	for (auto& kv : countMap)
	{
		cout << kv.first << ":" << kv.second << endl; 
			//西瓜:3
			//苹果 : 3
			//草莓 : 3
			//香蕉 : 1
	}
	cout << endl;
}

map(VS下)(有序的C P X X)

void map1()
{
	string arr[] = { "苹果","西瓜","草莓","苹果","西瓜","草莓" ,"苹果","西瓜","草莓" ,"香蕉" };
	map<string, int> countMap;
	for (auto& e : arr)
	{
		countMap[e]++;
	}
	for (auto& kv : countMap)
	{
		cout << kv.first << ":" << kv.second << endl;
		//草莓 : 3
		//苹果 : 3
		//西瓜 : 3
		//香蕉 : 1
	}
	cout << endl;
}

set和unordered_set性能测试:

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

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

在1000000有序数据下released下,set容器和unordered_set容器对1000000个数做增删查改操作所用的时间。

在1000000有序数据下debug下,set容器和unordered_set容器对1000000个数做增删查改操作所用的时间。

在1000000无序数据下debug下,set容器和unordered_set容器对1000000个数做增删查改操作所用的时间。

在1000000无序数据下released下,set容器和unordered_set容器对1000000个数做增删查改操作所用的时间。

int test_set2()
{
	const size_t N = 10000;

	unordered_set<int> us;
	set<int> s;

	vector<int> v;
	v.reserve(N);
	srand(time(0));
	for (size_t i = 0; i < N; ++i)
	{
		//v.push_back(rand()); // N比较大时,重复值比较多
		//v.push_back(rand()+i); // 重复值相对少
		v.push_back(i); // 没有重复,有序
	}

	size_t begin1 = clock();
	for (auto e : v)
	{
		s.insert(e);
	}
	size_t end1 = clock();
	cout << "set insert:" << end1 - begin1 << endl;

	size_t begin2 = clock();
	for (auto e : v)
	{
		us.insert(e);
	}
	size_t end2 = clock();
	cout << "unordered_set insert:" << end2 - begin2 << endl;

	int m1 = 0;
	size_t begin3 = clock();
	for (auto e : v)
	{
		auto ret = s.find(e);
		if (ret != s.end())
		{
			++m1;
		}
	}
	size_t end3 = clock();
	cout << "set find:" << end3 - begin3 << "->" << m1 << endl;

	int m2 = 0;
	size_t begin4 = clock();
	for (auto e : v)
	{
		auto ret = us.find(e);
		if (ret != us.end())
		{
			++m2;
		}
	}
	size_t end4 = clock();
	cout << "unorered_set find:" << end4 - begin4 << "->" << m2 << endl;

	cout << "插入数据个数:" << s.size() << endl;
	cout << "插入数据个数:" << us.size() << endl << endl;

	size_t begin5 = clock();
	for (auto e : v)
	{
		s.erase(e);
	}
	size_t end5 = clock();
	cout << "set erase:" << end5 - begin5 << endl;

	size_t begin6 = clock();
	for (auto e : v)
	{
		us.erase(e);
	}
	size_t end6 = clock();
	cout << "unordered_set erase:" << end6 - begin6 << endl << endl;

	return 0;
}

在千万级的时候set全方位落后于unordered_set

但也不完全是这样的,如果在有序的情况下,set还是很厉害的

  • 当处理数据量小时,map/set容器与unordered_map/unordered_set容器增删查改的效率差异不大。
  • 当处理数据量大时,map/set容器与unordered_map/unordered_set容器增删查改的效率相比,unordered系列容器的效率更高。
unordered_set和unordered_map都是C++标准库中的关联容器,它们的主要区别在于存储方式和元素的顺序性: 1. **存储方式**[^1]: - unordered_set内部使用哈希表来存储元素,每个元素都有一个唯一的哈希值,这使得插入、删除和查找操作的时间复杂度接近常数O(1),即使对于大型集合也是如此。 - unordered_map同样基于哈希表,但它存储的是键值对,键(Key)通过哈希函数映射到哈希桶,值(Value)存储在对应的位置。 2. **顺序性**: - mapset(包括unordered_set)默认按照元素的比较器(如默认情况下是自然排序)保持元素的顺序,但unordered_版本则不保证这一点,因为它们依赖于哈希函数的结果而不是元素本身的顺序。 3. **元素唯一性**: - 在unordered_set中,元素必须是唯一的,不允许有重复的值。 - unordered_map的键(Key)也是唯一的,如果试图插入已经存在的键,旧的值会被覆盖。 使用方法示例: ```cpp #include <unordered_set> #include <iostream> // 示例unordered_set std::unordered_set<int> unique_numbers; unique_numbers.insert(5); unique_numbers.insert(3); // 插入新元素,已存在5,所以不会重复插入 for (const auto& num : unique_numbers) { std::cout << num << " "; // 输出:3 5 } // 示例unordered_map std::unordered_map<std::string, int> name_to_age; name_to_age["Alice"] = 25; // 插入键值对 if (name_to_age.find("Bob") != name_to_age.end()) { std::cout << "Bob's age: " << name_to_age["Bob"] << "\n"; // 如果Bob存在,输出年龄 } else { std::cout << "Bob not found\n"; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

对自己好点儿i

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值