首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
C++ set容器去重与排序 _ insert函数与自定义比较器【实战】

C++ set容器去重与排序 _ insert函数与自定义比较器【实战】

热心网友
23
转载
2026-05-06

C++ set容器去重与排序:insert函数与自定义比较器实战解析

C++ set容器去重与排序 _ insert函数与自定义比较器【实战】

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

set插入重复元素时,如何准确判断insert是否成功?

判断C++ set插入操作是否成功,关键在于正确解读其返回值。标准库中的set::insert函数会返回一个std::pair类型的结果。其中,second成员是一个布尔标志:若为true,表明新元素已成功插入容器;若为false,则说明该键值已存在于集合中,本次插入操作被忽略。

这里需要特别注意一个常见错误:切勿通过判断返回的迭代器是否等于end()来确认插入结果。该迭代器始终指向一个有效元素——要么是新插入的,要么是集合中已存在的等价元素。因此,它无法用于判断重复。

例如,以下代码是无效的:

if (s.insert(x).first != s.end()) { ... } // 此条件恒成立,无实际意义

正确的判断方法如下:

auto [it, inserted] = s.insert(x); // C++17结构化绑定,简洁明了
if (inserted) {
    // 元素新增成功,可在此处理相关逻辑
} else {
    // 元素已存在,it指向集合中原有的对应元素
}

自定义比较器必须满足「严格弱序」,否则set行为未定义

许多C++ set容器出现的诡异崩溃或逻辑错误,根源往往在于自定义比较器违反了“严格弱序”的核心规则。该规则要求比较关系满足非自反性、反对称性和传递性。开发者常犯的错误是误用<=!=来实现比较逻辑。

以下是一个典型的错误示例:

struct BadComp {
    bool operator()(const int& a, const int& b) const {
        return a <= b; // ❌ 违反非自反性:a <= a 为 true,将导致未定义行为
    }
};

正确的做法是始终使用严格的<关系来定义比较逻辑,并确保其清晰无误。以下是几种常见场景的正确实现:

  • 按绝对值排序return abs(a) < abs(b);(需注意处理绝对值相等的不同数值)
  • 字符串先按长度、再按字典序比较return s1.size() < s2.size() || (s1.size() == s2.size() && s1 < s2);
  • 实现降序排列:直接使用return a > b;即可,避免使用!(a < b),后者在边界情况下可能破坏传递性。

想用set去重+排序,但又需要保留原始插入顺序?

答案是:set容器本身无法实现。set会根据你提供的比较器(或默认的less)对所有元素进行排序,原始插入顺序会被完全覆盖。如果你的核心需求是“去重但保持元素首次出现的顺序”,那么set并非合适的选择。即使尝试为元素添加时间戳字段,在多线程或重复值场景下,也难以保证稳定性和效率。

更实用的替代方案如下:

  • 使用std::unordered_set进行快速重复判断,同时配合std::vector按序存储唯一元素。典型代码模式为:if (seen.insert(x).second) unique_vec.push_back(x);
  • 若后续仍需对唯一序列进行快速查找,可将其封装为一个工具类,内部同时维护unordered_setvector
  • 切勿强行使用包含时间戳的自定义比较器来改造set。这会使排序逻辑复杂化,降低find等操作的效率,并可能因时间戳的重复或更新引发意外行为。

性能敏感场景:单次insert调用 vs 批量构造初始化

当需要插入大量元素时,性能差异将变得显著。逐个调用insert方法的时间复杂度为O(n log n)。而使用迭代器区间进行构造初始化(例如set s(v.begin(), v.end());),底层实现可能进行优化,虽然时间复杂度可能仍是O(n log n),但常数因子更小,更重要的是减少了多次内存分配的开销,提升了内存访问的局部性。

在以下实测场景中,性能差异较为明显:

  • 从vector批量去重排序:直接使用区间构造函数,通常比循环调用insert快10%到30%。
  • 数据基本有序时:可考虑使用std::set::insert的“提示”版本(即带迭代器参数作为插入位置提示)。若提示位置准确,插入的均摊时间复杂度可降至接近O(1);反之,性能将退化为普通的insert
  • 编译器优化:在-O2等优化级别下,GCC等编译器对于set{a,b,c}这类初始化列表,可能进行常量折叠等激进优化。

当然,如果插入操作分散在程序的不同逻辑分支中,则不必强行合并。代码的可读性和可维护性始终应放在首位。

最后,还有一个易被忽略的性能细节:自定义比较器的类型是set模板参数的一部分。若该比较器类型包含复杂的内部状态或较大的对象,可能会显著增加模板实例化的编译时间,以及最终二进制文件的体积。因此,保持比较器的轻量与简单,通常是一个良好的编程习惯。

来源:https://www.php.cn/faq/2313881.html
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

相关攻略

c++如何解析MPEG-TS流中的PAT与PMT节目表【深度】
编程语言
c++如何解析MPEG-TS流中的PAT与PMT节目表【深度】

C++如何解析MPEG-TS流中的PAT与PMT节目表【深度】 PAT表是解析MPEG-TS流的关键起点,它固定位于PID为0x0000的TS包中。解析时需通过payload_unit_start_indicator标志定位新表起始,正确处理adaptation field以找到payload,校验

热心网友
05.06
C++ std::identity用法 _ 函数对象占位符与ranges算法【详解】
编程语言
C++ std::identity用法 _ 函数对象占位符与ranges算法【详解】

C++ std::identity用法详解:函数对象占位符与ranges算法核心指南 std::identity 核心概念与应用场景解析 在C++20标准库中,std::identity绝非简单的语法糖,而是std::ranges算法体系中表达“元素原样透传”意图的唯一标准函数对象。当你调用std:

热心网友
05.06
C++ std::is_base_of用法 _ 编译期检查类继承关系【干货】
编程语言
C++ std::is_base_of用法 _ 编译期检查类继承关系【干货】

std::is_base_of编译期报错解析:非法类型、不完整类型与非类类型传入的应对方案 std::is_base_of 编译期报错的根本原因 许多C++开发者在首次使用 std::is_base_of 模板时,常对其在编译阶段直接报错感到困惑。这源于其作为类型特征(type trait)的本质—

热心网友
05.06
c++如何读取和设置文件的扩展时间戳信息_出生时间提取【技巧】
编程语言
c++如何读取和设置文件的扩展时间戳信息_出生时间提取【技巧】

Linux下birth time仅能通过statx()读取且不可设置,需内核≥4 11、支持的文件系统及正确挂载选项;glibc未暴露该字段,stat()等传统接口无法获取。 Linux 下用 stat 和 utimensat 读取 设置 birth time(创建时间) 在Linux的世界里,文件

热心网友
05.06
c++ cista++序列化 c++如何进行极低延迟的对象序列化
编程语言
c++ cista++序列化 c++如何进行极低延迟的对象序列化

cista 实现微秒级序列化的核心原理:零开销内存拷贝与偏移重定位 cista 微秒级序列化的技术实现解析 cista 之所以能够实现微秒甚至纳秒级的序列化性能,源于其颠覆性的设计理念。与传统的序列化方案不同,cista 彻底摒弃了运行时类型识别(RTTI)、动态反射和堆内存分配等重型操作。它采用了

热心网友
05.06

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

荣耀400pro关机要按几秒
电脑教程
荣耀400pro关机要按几秒

荣耀400 Pro正确关机全指南:从常规操作到故障应对详解 需要关闭您的荣耀400 Pro手机?日常操作其实非常简便。只需长按位于机身右侧的电源键约3秒钟,屏幕上便会浮现一个简洁的半透明菜单,其中明确列出了“关机”、“重启”以及“紧急呼叫”选项。直接点击“关机”,系统将启动一次10秒的安全倒计时,随

热心网友
05.06
红米K30Pro如何拆后盖胶怎么清理
电脑教程
红米K30Pro如何拆后盖胶怎么清理

红米K30 Pro后盖拆解教程:专业工具与细致手法的完美结合 红米K30 Pro的后盖采用了高强度背胶配合隐藏式螺丝的双重固定设计,想要实现无损拆解,绝非依靠蛮力可以完成。整个操作流程对加热温度、撬启手法以及清洁标准都有严格要求,任何环节的疏忽都可能导致部件损伤。具体而言,其后盖边缘使用了耐高温的工

热心网友
05.06
三星zflip电池百分比需要root吗
电脑教程
三星zflip电池百分比需要root吗

无需Root权限:三星Galaxy Z Flip系列电量数字显示设置全解析 很多三星折叠屏手机用户都想知道,如何在状态栏直接查看精确的电池百分比数字,是否必须获取Root权限才能实现?实际上完全不需要。三星自Galaxy Z Flip 5、Z Flip 4等主流机型开始,已在系统层面内置了这一实用功

热心网友
05.06
笔记本开机自检时能看到DDR3或DDR4吗
电脑教程
笔记本开机自检时能看到DDR3或DDR4吗

笔记本开机自检信息虽不直接标注“DDR3”或“DDR4”,但联想、戴尔、华硕等品牌BIOS画面常以“PC3-”或“PC4-”编码间接揭示内存代际。UEFI自检显示的内存频率(如2400MHz 3200MHz)结合JEDEC规范可辅助推断:PC3对应DDR3,PC4对应DDR4。更高精度的识别方案包括

热心网友
05.06
空调制冷但不太凉是压缩机问题吗?
电脑教程
空调制冷但不太凉是压缩机问题吗?

空调制冷不足怎么办?先别急着维修压缩机,这些问题更常见 夏天开空调却感觉不够凉爽?很多朋友的第一反应是压缩机坏了,其实压缩机故障的概率相对较低。根据维修行业的大数据统计,绝大多数制冷效果不佳的情况,源于几个容易被忽略的日常维护与环境因素。滤网积尘、制冷剂泄漏、外机散热不良才是真正的高发原因。盲目更换

热心网友
05.06