0

0

c++如何实现LRU Cache缓存机制_c++ 双向链表与哈希映射结合【案例】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2025-12-30 13:53:02

|

132人浏览过

|

来源于php中文网

原创

必须协同使用std::list和std::unordered_map:list维护访问顺序且迭代器稳定(删除非所指节点不失效),map提供o(1)定位;capacity为0时put应直接返回。

c++如何实现lru cache缓存机制_c++ 双向链表与哈希映射结合【案例】

为什么不能只用 std::list 或只用 std::unordered_map

单独用 std::list 无法在 O(1) 时间定位某个 key 对应的节点——每次 get 都得遍历,退化成 O(n);只用 std::unordered_map 则无法维护访问顺序,淘汰最久未用项时得遍历所有 value 找最小时间戳,也是 O(n)。必须让两者协作:unordered_map 存 key → list::iterator 映射,list 存实际 pair<int int></int> 并保证头尾有序。

list::iterator 能稳定指向元素的前提是什么

std::list 的迭代器在插入/删除其他节点时不失效(这是它和 vector 的关键区别),但前提是不删除该迭代器本身指向的节点。所以你在 get 时把节点移到 front,或在 put 时删除 back 节点,都必须先用 map 查到对应 iterator,再用 spliceerase 操作——不能直接保存裸指针或索引。

  • 错误写法:auto ptr = &(*it); —— list 迭代器不是指针,取地址无意义且不保活
  • 正确做法:cache_list.splice(cache_list.begin(), cache_list, it); 移动节点
  • map 中存储的是 std::list<pair>>::iterator</pair>,不是 pair*

如何避免 put 时重复插入导致内存泄漏或逻辑错乱

常见坑是:key 已存在,却先 erase map 项、再 push_front 新节点,最后再 insert map —— 这中间如果 list 操作失败(比如内存不足),map 就丢了映射,且旧节点还在 list 里。更安全的做法是统一用「查-删-插」流程,并复用原节点:

遨虾
遨虾

1688推出的跨境电商AI智能体

下载
void put(int key, int value) {
    if (cache_map.find(key) != cache_map.end()) {
        auto it = cache_map[key];
        it->second = value;                    // 更新值
        cache_list.splice(cache_list.begin(), cache_list, it); // 移至头部
    } else {
        cache_list.push_front({key, value});
        cache_map[key] = cache_list.begin();
        if (cache_map.size() > capacity) {
            int tail_key = cache_list.back().first;
            cache_map.erase(tail_key);
            cache_list.pop_back();
        }
    }
}

构造函数里 capacity 为 0 怎么处理

不少实现忽略边界情况,导致 capacity == 0put 后立即删光,get 永远返回 -1。应该在 put 开头加判断:

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

  • capacity == 0,直接 return(不存任何数据)
  • capacity ,按 0 处理(防御性编程)
  • 注意 cache_map.max_size() 和实际容量无关,别混淆

链表和哈希表的生命周期必须完全一致:list 析构时所有 iterator 自动失效,map 里存的 iterator 不会自动清空,所以类析构不需要显式清理 iterator,但必须确保 list 在 map 之后析构(成员声明顺序要 list 在前、map 在后,或用 RAII 管理)。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

910

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

596

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

294

2025.08.29

C++中int的含义
C++中int的含义

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

210

2025.08.29

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

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

77

2025.09.05

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

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

39

2025.11.16

golang map原理
golang map原理

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

67

2025.11.17

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

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

47

2025.11.27

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

48

2026.02.28

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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