【STL】set、multiset、unordered_set、unordered_multiset 的区别

在 C++ STL 中,setmultiset 是经常使用的关联式容器,而从 C++11 开始,又引入了 unordered_setunordered_multiset

很多同学在学习时常常会混淆它们的区别,今天我们就来系统梳理一下。


1. set

  • 特点

    • 元素 唯一
    • 元素 自动排序(默认升序,可自定义比较函数)
    • 底层实现:红黑树(平衡二叉搜索树)
  • 示例

c++ 复制代码
#include <set>
#include <iostream>
using namespace std;

int main() {
    set<int> s;
    s.insert(3);
    s.insert(1);
    s.insert(3); // 插入失败,元素必须唯一
    for (int v : s) cout << v << " "; // 1 3
}

2. multiset

  • 特点

    • 元素 允许重复
    • 元素 自动排序
    • 底层实现:红黑树
  • 示例

c++ 复制代码
#include <set>
#include <iostream>
using namespace std;

int main() {
    multiset<int> ms;
    ms.insert(3);
    ms.insert(1);
    ms.insert(3); // 插入成功,可重复
    for (int v : ms) cout << v << " "; // 1 3 3
}
  • 删除操作差异

    c++ 复制代码
    multiset<int> ms = {1, 2, 2, 3};
    ms.erase(2);  // 删除所有 key=2 的元素
    // 如果只想删除一个,可以用迭代器:
    auto it = ms.find(2);
    if (it != ms.end()) ms.erase(it);

3. unordered_set

  • 特点

    • 元素 唯一
    • 元素 无序存储(不保证顺序)
    • 底层实现:哈希表
    • 查找、插入、删除 平均 O(1) ,性能通常比 set 更高
  • 示例

c++ 复制代码
#include <unordered_set>
#include <iostream>
using namespace std;

int main() {
    unordered_set<int> us;
    us.insert(5);
    us.insert(1);
    us.insert(5); // 插入失败,元素必须唯一
    for (int v : us) cout << v << " "; // 输出顺序不固定
}

4. unordered_multiset

  • 特点

    • 元素 允许重复
    • 元素 无序存储
    • 底层实现:哈希表
    • 查找、插入、删除 平均 O(1)
  • 示例

c++ 复制代码
#include <unordered_set>
#include <iostream>
using namespace std;

int main() {
    unordered_multiset<int> ums;
    ums.insert(5);
    ums.insert(5);
    ums.insert(1);
    for (int v : ums) cout << v << " "; // 输出顺序不固定
}

5. 四者对比总结

其实这四者的区别很明显也很容易懂。只要不去涉及底层原理,看下面这张表就行了。

容器类型 元素唯一性 是否有序 底层结构 插入/查找/删除复杂度
set 唯一 有序 红黑树 O(log n)
multiset 可重复 有序 红黑树 O(log n)
unordered_set 唯一 无序 哈希表 平均 O(1),最坏 O(n)
unordered_multiset 可重复 无序 哈希表 平均 O(1),最坏 O(n)

总结一下

就是带有multi的就是可以有重复元素;带有unordered就是无序,然后性能更好点。


6. 使用场景的建议

  • set :需要保持元素唯一 ,且要有序(如 ID 集合、排行榜)。
  • multiset :需要存储重复元素 ,且要有序(如成绩表、频率统计)。
  • unordered_set :需要元素唯一 ,且更关心性能,不在意顺序(如哈希去重)。
  • unordered_multiset :需要存储重复元素 ,且更关心性能,不在意顺序(如计数场景)。

相关推荐
Vect__7 小时前
基于线程池从零实现TCP计算器网络服务
c++·网络协议·tcp/ip
草履虫建模10 小时前
力扣算法 1768. 交替合并字符串
java·开发语言·算法·leetcode·职场和发展·idea·基础
naruto_lnq12 小时前
分布式系统安全通信
开发语言·c++·算法
Jasmine_llq13 小时前
《P3157 [CQOI2011] 动态逆序对》
算法·cdq 分治·动态问题静态化+双向偏序统计·树状数组(高效统计元素大小关系·排序算法(预处理偏序和时间戳)·前缀和(合并单个贡献为总逆序对·动态问题静态化
爱吃rabbit的mq13 小时前
第09章:随机森林:集成学习的威力
算法·随机森林·集成学习
(❁´◡`❁)Jimmy(❁´◡`❁)14 小时前
Exgcd 学习笔记
笔记·学习·算法
YYuCChi14 小时前
代码随想录算法训练营第三十七天 | 52.携带研究材料(卡码网)、518.零钱兑换||、377.组合总和IV、57.爬楼梯(卡码网)
算法·动态规划
CSDN_RTKLIB14 小时前
【四个场景测试】源文件编码UTF-8 BOM
c++
不能隔夜的咖喱15 小时前
牛客网刷题(2)
java·开发语言·算法
VT.馒头15 小时前
【力扣】2721. 并行执行异步函数
前端·javascript·算法·leetcode·typescript