对于web开发而言,缓存必不可少,也是提高性能最常用的方式。无论是浏览器缓存(如果是chrome浏览器,可以通过chrome:://cache查看),还是服务端的缓存(通过memcached或者redis等内存数据库)。缓存不仅可以加速用户的访问,同时也可以降低服务器的负载和压力。那么,了解常见的缓存淘汰算法的策略和原理就显得特别重要。像浏览器的缓存策略、memcached的缓存策略都是使用LRU这个算法,LRU算法会将近期最不会访问的数据淘汰掉。LRU如此流行的原因是实现比较简单,而且对于实际问题也很实用,良好的运行时性能,命中率较高。下面谈谈如何实现LRU缓存:

常见的缓存算法

  • LRU (Least recently used) 最近最少使用,如果数据最近被访问过,那么将来被访问的几率也更高。
  • LFU (Least frequently used) 最不经常使用,如果一个数据在最近一段时间内使用次数很少,那么在将来一段时间内被使用的可能性也很小。
  • 除此之外还有FIFO、随机算法等。

实现思路

LRU算法实现并不难,但是要高效地实现却是有难度的,要想高效实现其中的插入、删除、查找,第一想法就是红黑树,但是红黑树也是一种折中的办法。插入、删除效率最高当属链表,查找效率当属hash。所以完全有理由想到将双向链表+hashtable的方案,利用空间换时间,实现LRU算法。具体来说其主要包括set和get两个操作。

其基本数据结构就是一个链表:

  • 新数据插入时是插入到链表头部;
  • 每当缓存命中(即缓存数据被访问),就将这个数据所在的节点移到链表头部;
  • 当链表满的时候,从链表尾部开始丢弃数据(节点)。

LRU Cache无非set/get两种操作:

  • 1、set(key,value):1)如果key在hashmap中存在,则先重置对应的value值(一个key只能对应一个节点/value),然后将该节点移动到链表的头部。2)如果key在hashmap不存在,则新建一个节点,并将节点放到链表的头部。当Cache存满的时候,将链表最后一个节点删除即可。
  • 2、get(key):如果key在hashmap中存在,则把对应的节点放到链表头部,并返回对应的value值;如果不存在,则返回-1。

注:其基本结构就是一个链表,但是考虑到仅使用链表会存在查找效率低下的问题所以引入hashmap重新的对key-value冗余了一份。这样就兼顾了动态与静态的操作。

废话少说首先看代码:

class LRUCache {
private://LRU数据结构struct Node {int key;int value;Node(int k, int v) :key(k), value(v) {}};
public:LRUCache(int c) :capacity(c) {}int get(int key) {if (cacheMap.find(key) == cacheMap.end())return -1; //这里产生缺页中断,根据页表将页面调入内存,然后set(key, value)//将key移到第一个,并更新cacheMap cacheList.splice(cacheList.begin(), cacheList, cacheMap[key]);cacheMap[key] = cacheList.begin();return cacheMap[key]->value;}void set(int key, int value) {if (cacheMap.find(key) == cacheMap.end()){//如果已经满了那么就将最后一个淘汰掉,然后将该其加到第一个位置if (cacheList.size() == capacity){cacheMap.erase(cacheList.back().key);cacheList.pop_back();}cacheList.push_front(Node(key, value));cacheMap[key] = cacheList.begin();//确实一个key对应一个双向链表}else{//更新节点的值,并将其加到第一个位置cacheMap[key]->value = value;cacheList.splice(cacheList.begin(), cacheList, cacheMap[key]);cacheMap[key] = cacheList.begin();}}private://最大可以容纳的节点数(key-value)对。一个key只能对应一个节点。int capacity;//自始至终都只有这么一个链表list<Node> cacheList;//cacheMap相当于以另外一种形式来存储key-value。是的查询效率变为O(1)。unordered_map<int, list<Node>::iterator> cacheMap;
};

对于新手来说有以下几点需要注意:

(1)一个key只会有一个对应的value会被保存下来,不会有同一个key的多个节点被保留下来。
(2)链表始终只有一个,而且代码中的容量capbility就是指链表的长度也就是key的个数。
(3)链表是真实存在的,只不过每个key-value对还会以hashmap的形式存储。正如前面所述的那样兼顾了动态的插入删除操作和静态的查找操作。

C++实现LRU(Least-Recently Used)缓存算法相关推荐

  1. 操作系统之虚拟存储管理 java python 实现 最优(Optimal)置换算法 先进先出(FIFO)页面置换算法 LRU(Least Recently Used)置换算法

    操作系统之虚拟存储管理 实验内容:模拟请求分页虚拟存器管理技术中的硬件地址变换.缺页中断以及页式淘汰算法,处理缺页中断. 实验目的:清楚认识请求分页管理. 最佳(Optimal)置换算法 其所选择的被 ...

  2. LRU(Least Recently Used)缓存淘汰策略算法

    LRU LRU(Least Recently Used,最近最少使用)算法根据数据的历史访问记录来进行淘汰数据,其核心思想是"如果数据最近被访问过,那么将来被访问的几率也更高". ...

  3. 保研面试 算法题_面试挂在了一道 LRU 缓存算法设计题

    前言 好吧,有人可能觉得我标题党了,但我想告诉你们的是,前阵子面试确实挂在了 RLU 缓存算法的设计上了. 当时面试官问到这个题的时候,很快给了一个思路,但是手写的时候,发现自己没有深刻理解它,加上当 ...

  4. android lru进程队列,Android缓存机制——LruCache

    一.Android中的缓存策略 Android的缓存,主要的就是内存缓存和硬盘缓存. 不管是内存缓存还是硬盘缓存,它们的缓存大小都是有限的.当缓存满了之后,再想其添加缓存,这个时候就需要删除一些旧的缓 ...

  5. 【文科生带你读JavaScript数据结构与算法】2. 双向链表与LRU缓存算法原理与实现(下)

    上篇聊了聊双向链表(doubly linked list)这个数据结构,今天就来点更实际的,也可以用双链表来实现的一种生活工作中不可或缺的优化算法--LRU缓存(Least Recently Used ...

  6. 缓存算法:LFU和LRU的JAVA实现

    LFU.LRU的实现总结 LFU和LRU均是常见的缓存算法,其中LFU表示最不经常使用,LRU表示最近最少使用. 10.1 LRU(Least Recently Used) 定义 LRU (最近最少使 ...

  7. 最近最少使用LRU(Least Recently Used)算法java实现

    最近最少使用LRU(Least Recently Used)算法java实现 一.使用LinkedHashMap算法实现 二.手撸 LRU 算法实现(Hash表 + 双向链表) 三.总结 最近最少使用 ...

  8. LRU(least recently used)算法浅析

    LRU(Least recently used)算法,顾名思义:最近最少使用. LRU-1算法 算法根据数据的历史访问记录来进行淘汰数据,其核心思想是"如果数据最近被访问过,那么将来被访问的 ...

  9. 实现 LRU 缓存算法

    1 LRU 缓存介绍 LRU 算法全称是最近最少使用算法(Least Recently Use),是一种简单的缓存策略.顾名思义,LRU 算法会选出最近最少使用的数据进行淘汰. 那么什么是缓存呢?缓存 ...

  10. 缓存算法(FIFO 、LRU、LFU三种算法的区别)

    FIFO算法 FIFO 算法是一种比较容易实现的算法.它的思想是先进先出(FIFO,队列),这是最简单.最公平的一种思想,即如果一个数据是最先进入的,那么可以认为在将来它被访问的可能性很小.空间满的时 ...

最新文章

  1. Hyper-V 故障转移群集管理
  2. 构造函数与析构函数到底做了啥?
  3. Ansible之使用角色一键部署httpd并检查部署结果
  4. Android AIDL使用介绍(3) 浅说AIDL背后的Binder
  5. 算法函数_关于损失函数和优化算法,看这一篇就够了
  6. Java技术栈---语言基础
  7. 外行人都能看懂的SpringCloud
  8. 通俗易懂!视觉slam第十一部分——线性系统和卡尔曼滤波
  9. latex缩进与对齐_科学网—Latex算法排版,不同行缩进不同 - 张鹏的博文
  10. 2006年百度之星程序设计大赛试题初赛题目-题6-百度语言翻译机
  11. SpringBoot配置多数据源(动态切换)
  12. 三相坐标变换CLARK变换PARK变换
  13. 基于ssm的超市进销存管理系统
  14. 前程无忧网站大数据职位信息分析可视化(源码)
  15. 力扣刷题 DAY_81 贪心
  16. 软件-未能加载文件或程序集.HRESULT:0x80131515解决方法
  17. hdu1546—Idiomatic Phrases Game(spfa)
  18. 简易PC基准测试丨PerformanceTest功能简介
  19. 我是如何拿到:百度 腾讯 头条 美团 度小满等互联网offer的?
  20. 指纹算法中控 X638考勤机编程(delphi)

热门文章

  1. java restful文件传输_Spring Boot 2.0实现基于Restful风格的文件上传与下载APIs
  2. PTA 程序设计天梯赛(121~140题)
  3. 《高效对话》— 综合素质提升书籍
  4. Android 实现 欢迎界面 自动跳转 到 主界面
  5. vue.js 源代码学习笔记 ----- instance inject
  6. composer 的快速安装
  7. printf函数重定向
  8. Snabbt.js – 极简的 JavaScript 动画库
  9. ORACLE 發送郵件(全)
  10. DbEntry.Net---又一个国产开源ORM数据访问及 WEB 框架