0

0

C++如何实现高性能LRU缓存_C++结合哈希表与双向链表底层实现【进阶】

穿越時空

穿越時空

发布时间:2026-02-07 12:58:02

|

648人浏览过

|

来源于php中文网

原创

不能只用 std::unordered_map 实现 LRU,因其不维护访问顺序,无法 O(1) 定位最近最少使用节点;必须结合双向链表实现节点位置更新与淘汰,哈希表存 Node* 以支持零拷贝指针操作,并通过内存池或 deque 复用节点避免频繁分配。

c++如何实现高性能lru缓存_c++结合哈希表与双向链表底层实现【进阶】

为什么不能只用 std::unordered_map 实现 LRU?

因为 std::unordered_map 本身不维护访问顺序,无法在 O(1) 时间内定位“最近最少使用”的节点。LRU 的核心诉求是:每次 getput 都要更新节点位置,且淘汰时取尾部——这必须靠双向链表支撑;而哈希表只负责 O(1) 定位节点指针。两者缺一不可。

常见错误是试图用 std::map 按时间戳排序,或给每个键存一个 last_access_time 再遍历找最小值——这两种都退化到 O(n),完全违背 LRU 的高性能初衷。

std::list 和自定义双向链表哪个更合适?

std::list 看似省事,但存在两个硬伤:一是迭代器失效风险(如外部代码意外调用 list::spliceerase);二是它不支持直接把某个节点移到头部——你得先 erasepush_front,触发两次内存操作,且 erase 后原迭代器立即失效,无法安全持有。

更稳妥的做法是手写轻量级双向链表节点:

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

struct Node {
    int key, value;
    Node* prev;
    Node* next;
    Node(int k, int v) : key(k), value(v), prev(nullptr), next(nullptr) {}
};

这样能用 move_to_head() 原地调整指针,全程无内存分配、无迭代器失效,真正满足 O(1) 要求。

WOMBO
WOMBO

使用AI创作美丽的艺术品

下载

哈希表该存什么?int 还是 Node*

必须存 Node*,不是 intstd::shared_ptr。原因很实际:

  • std::unordered_map 查找后直接拿到指针,后续所有链表操作(断开、重连、删除)都基于该地址,零拷贝
  • std::shared_ptr 会引入原子计数开销,且缓存频繁读写时引用计数竞争反而成瓶颈
  • int(比如节点索引)意味着还要额外维护一个 std::vector,不仅破坏局部性,还让扩容时指针全部失效

注意:Node* 是裸指针,但只要保证节点生命周期由缓存类完全控制(例如全放在 std::vector 池里 or new/delete 配对),就完全可控。

如何避免 put 时重复构造/析构 Node

高频缓存场景下,反复 new / delete 是性能杀手。推荐两种实操方案:

  • std::vector 预分配固定大小内存池,配合空闲链表管理可用节点(free_list_head 指向首个空闲节点)
  • 或直接用 std::deque —— 它的内存分段连续,push_front/pop_back 不导致整体迁移,且支持 O(1) 随机访问和稳定指针

无论哪种,都要重载 LRUCache::get()LRUCache::put() 中的节点复用逻辑:命中时仅改值+移位;未命中且满时,复用尾部节点内存,而不是 delete + new。

真正难的不是结构设计,而是把哈希查找、指针解引用、内存复用这三步压进一条 CPU 流水线里——稍有不慎,cache line 就频繁 miss,再好的算法也跑不满带宽。

相关文章

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载

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

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

606

2023.08.02

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

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

551

2024.08.29

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

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

173

2025.08.29

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

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

204

2025.08.29

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相关内容,阅读专题下面的文章了解更多详细内容。

64

2025.11.17

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

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

43

2025.11.27

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

2

2026.02.06

热门下载

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

精品课程

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

共102课时 | 6.9万人学习

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

共162课时 | 19.5万人学习

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

共119课时 | 12.7万人学习

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

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