0

0

c++中如何实现简单的LRU淘汰策略_c++双向链表与哈希表结合【详解】

尼克

尼克

发布时间:2026-01-18 14:40:49

|

977人浏览过

|

来源于php中文网

原创

必须用双向链表+哈希表组合:双向链表支持O(1)节点移动与删除,哈希表提供key到节点迭代器的O(1)索引;单独list无法O(1)定位,单独unordered_map无法维护时序。

c++中如何实现简单的lru淘汰策略_c++双向链表与哈希表结合【详解】

为什么必须用双向链表 + 哈希表组合

单靠 std::list 无法在 O(1) 时间定位某个 key 对应的节点位置;单靠 std::unordered_map 又无法维护访问时序。LRU 的核心操作——“访问即置顶”和“淘汰尾部”——要求:查找快(O(1))、移动快(O(1))、删除快(O(1))。只有双向链表支持任意节点的常数时间摘除与插入,而哈希表提供 key 到链表节点指针的直接映射。

如何设计节点与容器结构

不能直接存 std::list<:pair v>>,因为 erase() 需要迭代器,而哈希表里得存这个迭代器。正确做法是:哈希表值类型为 std::list::iterator,Node 为自定义结构体,含 keyvalue。这样每次访问 key 时,能用哈希表 O(1) 拿到对应节点的迭代器,再用 splice() 把它移到链表头部。

  • std::list 存储所有活跃项,头为最近使用,尾为最久未用
  • std::unordered_map::iterator> 提供 key → 节点位置的直连索引
  • 每次 get()put() 都要调用私有函数 moveToHead(iterator),用 splice(head_iter, list, it) 实现零拷贝移动

put() 中淘汰逻辑怎么写才不漏判

淘汰只发生在 put() 且容量超限时,但要注意两种情况都得处理:key 已存在(更新 value 并刷新位置),key 不存在(插入新节点并可能淘汰)。容易出错的是:先 insert 再检查 size,导致实际 size 超过 capacity + 1 才删——这会让缓存短暂溢出。

void put(int key, int value) {
    if (cache.find(key) != cache.end()) {
        cache[key]->value = value;
        moveToHead(cache[key]);
        return;
    }
    Node node{key, value};
    lru_list.push_front(node);
    cache[key] = lru_list.begin(); // 迭代器有效
    if (cache.size() > capacity) {
        int tail_key = lru_list.back().key;
        lru_list.pop_back();
        cache.erase(tail_key); // 必须先删链表再删 map,否则迭代器失效
    }
}

迭代器失效是最大陷阱

std::list 的迭代器只有在对应节点被 erase() 时才失效;push_front()splice()pop_back() 都不使其他迭代器失效。但一旦你对某个节点调用了 erase(),所有指向它的迭代器立刻变悬垂指针。所以 cache.erase(tail_key) 必须在 lru_list.pop_back() 之后——否则你拿着一个刚被删节点的迭代器还试图从 map 里删它,行为未定义。

Magic Write
Magic Write

Canva旗下AI文案生成器

下载

立即学习C++免费学习笔记(深入)”;

另一个常见错误:用 auto it = lru_list.begin(); lru_list.erase(it); 后继续用 it ——哪怕只是打印,也属于未定义行为。所有涉及 erase 的地方,务必确认后续不再使用该迭代器。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

196

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

189

2025.07.04

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

75

2025.09.05

golang map相关教程
golang map相关教程

本专题整合了golang map相关教程,阅读专题下面的文章了解更多详细内容。

36

2025.11.16

golang map原理
golang map原理

本专题整合了golang map相关内容,阅读专题下面的文章了解更多详细内容。

59

2025.11.17

java判断map相关教程
java判断map相关教程

本专题整合了java判断map相关教程,阅读专题下面的文章了解更多详细内容。

37

2025.11.27

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

42

2026.01.16

全民K歌得高分教程大全
全民K歌得高分教程大全

本专题整合了全民K歌得高分技巧汇总,阅读专题下面的文章了解更多详细内容。

74

2026.01.16

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

23

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
HTML5/CSS3/JavaScript/ES6入门课程
HTML5/CSS3/JavaScript/ES6入门课程

共102课时 | 6.7万人学习

前端基础到实战(HTML5+CSS3+ES6+NPM)
前端基础到实战(HTML5+CSS3+ES6+NPM)

共162课时 | 18.9万人学习

第二十二期_前端开发
第二十二期_前端开发

共119课时 | 12.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号