`
uule
  • 浏览: 6358883 次
  • 性别: Icon_minigender_1
  • 来自: 一片神奇的土地
社区版块
存档分类
最新评论

缓存淘汰算法系列之1 --- LRU

 
阅读更多

1. LRU
1.1. 原理

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

1.2. 实现

最常见的实现是使用一个链表保存缓存数据,详细算法实现如下:

1. 新数据插入到链表头部;

2. 每当缓存命中(即缓存数据被访问),则将数据移到链表头部;

3. 当链表满的时候,将链表尾部的数据丢弃。

1.3. 分析

【命中率】

当存在热点数据时,LRU的效率很好,但偶发性的、周期性的批量操作会导致LRU命中率急剧下降,缓存污染情况比较严重。

【复杂度】

实现简单。

【代价】

命中时需要遍历链表,找到命中的数据块索引,然后需要将数据移到头部。

 

2. LRU-K(描述有误,请勿参考)

2.1. 原理

LRU-K中的K代表最近使用的次数,因此LRU可以认为是LRU-1。LRU-K的主要目的是为了解决LRU算法“缓存污染”的问题,其核心思想是将“最近使用过1次”的判断标准扩展为“最近使用过K次”。

2.2. 实现

相比LRU,LRU-K需要多维护一个队列,用于记录所有缓存数据被访问的历史。只有当数据的访问次数达到K次的时候,才将数据放入缓存。当需要淘汰数据时,LRU-K会淘汰第K次访问时间距当前时间最大的数据。详细实现如下:

1. 数据第一次被访问,加入到访问历史列表;

2. 如果数据在访问历史列表里后没有达到K次访问,则按照一定规则(FIFO,LRU)淘汰;

3. 当访问历史队列中的数据访问次数达到K次后,将数据索引从历史队列删除,将数据移到缓存队列中,并缓存此数据,缓存队列重新按照时间排序;

4. 缓存数据队列中被再次访问后,重新排序;

5. 需要淘汰数据时,淘汰缓存队列中排在末尾的数据,即:淘汰“倒数第K次访问离现在最久”的数据。

LRU-K具有LRU的优点,同时能够避免LRU的缺点,实际应用中LRU-2是综合各种因素后最优的选择,LRU-3或者更大的K值命中率会高,但适应性差,需要大量的数据访问才能将历史访问记录清除掉。

2.3. 分析

【命中率】

LRU-K降低了“缓存污染”带来的问题,命中率比LRU要高。

【复杂度】

LRU-K队列是一个优先级队列,算法复杂度和代价比较高。

【代价】

由于LRU-K还需要记录那些被访问过、但还没有放入缓存的对象,因此内存消耗会比LRU要多;当数据量很大的时候,内存消耗会比较可观。

LRU-K需要基于时间进行排序(可以需要淘汰时再排序,也可以即时排序),CPU消耗比LRU要高。

3. Two queues(2Q)

3.1. 原理

Two queues(以下使用2Q代替)算法类似于LRU-2,不同点在于2Q将LRU-2算法中的访问历史队列(注意这不是缓存数据的)改为一个FIFO缓存队列,即:2Q算法有两个缓存队列,一个是FIFO队列,一个是LRU队列

3.2. 实现

当数据第一次访问时,2Q算法将数据缓存在FIFO队列里面,当数据第二次被访问时,则将数据从FIFO队列移到LRU队列里面,两个队列各自按照自己的方法淘汰数据。详细实现如下:

1. 新访问的数据插入到FIFO队列;

2. 如果数据在FIFO队列中一直没有被再次访问,则最终按照FIFO规则淘汰;

3. 如果数据在FIFO队列中被再次访问,则将数据移到LRU队列头部;

4. 如果数据在LRU队列再次被访问,则将数据移到LRU队列头部;

5. LRU队列淘汰末尾的数据。

 

注:上图中FIFO队列比LRU队列短,但并不代表这是算法要求,实际应用中两者比例没有硬性规定。

3.3. 分析

【命中率】

2Q算法的命中率要高于LRU。

【复杂度】

需要两个队列,但两个队列本身都比较简单。

【代价】

FIFO和LRU的代价之和。

2Q算法和LRU-2算法命中率类似,内存消耗也比较接近,但对于最后缓存的数据来说,2Q会减少一次从原始存储读取数据或者计算数据的操作。

4. Multi Queue(MQ)

4.1. 原理

MQ算法根据访问频率将数据划分为多个队列,不同的队列具有不同的访问优先级,其核心思想是:优先缓存访问次数多的数据

4.2. 实现

MQ算法将缓存划分为多个LRU队列,每个队列对应不同的访问优先级。访问优先级是根据访问次数计算出来的,例如

详细的算法结构图如下,Q0,Q1....Qk代表不同的优先级队列,Q-history代表从缓存中淘汰数据,但记录了数据的索引和引用次数的队列:

 

如上图,算法详细描述如下:

1. 新插入的数据放入Q0;

2. 每个队列按照LRU管理数据;

3. 当数据的访问次数达到一定次数,需要提升优先级时,将数据从当前队列删除,加入到高一级队列的头部;

4. 为了防止高优先级数据永远不被淘汰,当数据在指定的时间里访问没有被访问时,需要降低优先级,将数据从当前队列删除,加入到低一级的队列头部;

5. 需要淘汰数据时,从最低一级队列开始按照LRU淘汰;每个队列淘汰数据时,将数据从缓存中删除,将数据索引加入Q-history头部;

6. 如果数据在Q-history中被重新访问,则重新计算其优先级,移到目标队列的头部;

7. Q-history按照LRU淘汰数据的索引。

4.3. 分析

【命中率】

MQ降低了“缓存污染”带来的问题,命中率比LRU要高。

【复杂度】

MQ需要维护多个队列,且需要维护每个数据的访问时间,复杂度比LRU高。

【代价】

MQ需要记录每个数据的访问时间,需要定时扫描所有队列,代价比LRU要高。

注:虽然MQ的队列看起来数量比较多,但由于所有队列之和受限于缓存容量的大小,因此这里多个队列长度之和和一个LRU队列是一样的,因此队列扫描性能也相近。

 

5. LRU类算法对比

由于不同的访问模型导致命中率变化较大,此处对比仅基于理论定性分析,不做定量分析。

对比点

对比

命中率

LRU-2 > MQ(2) > 2Q > LRU

复杂度

LRU-2 > MQ(2) > 2Q > LRU

代价

LRU-2  > MQ(2) > 2Q > LRU

实际应用中需要根据业务的需求和对数据的访问情况进行选择,并不是命中率越高越好。例如:虽然LRU看起来命中率会低一些,且存在”缓存污染“的问题,但由于其简单和代价小,实际应用中反而应用更多。

 

来源:http://blog.csdn.net/yunhua_lee/article/details/7599671

分享到:
评论

相关推荐

    缓存淘汰算法之LRU

    缓存淘汰算法之 LRU 缓存淘汰算法是指在计算机系统中,为了提高缓存命中率和减少缓存 pollution 而采用的算法。其中,LRU(Least Recently Used,最近最少使用)算法是一种常用的缓存淘汰算法。 1. LRU 算法原理 ...

    cache缓存淘汰算法--LRU算法.docx

    这些缓存淘汰算法各有优劣,LRU 实现简单但可能受到批量操作的影响,LRU-K 提高了命中率但成本较高,2Q 在两者之间找到了平衡,而 MQ 则提供了更灵活的策略以适应不同场景的需求。选择哪种算法取决于具体的应用场景...

    cache缓存淘汰算法--LRU算法.pdf

    综上所述,不同的缓存淘汰算法各有优劣,选择哪种算法取决于具体的应用场景和资源限制。LRU适合简单快速的实现,2Q在性能和复杂度之间找到了平衡,而LRU-K和MQ则提供了更高的灵活性和命中率,但代价也更高。在实际...

    cache缓存淘汰算法--LRU算法 (2).pdf

    LRU(Least Recently Used)缓存淘汰算法是根据数据的历史访问记录来决定何时淘汰数据的策略。其核心理念是“最近被访问过的数据在未来更有可能被再次访问”。LRU算法通常通过链表来实现,新数据插入链表头部,每次...

    cache缓存淘汰算法--LRU算法 (2).docx

    LRU-K算法改进了LRU的命中率,但其复杂度和代价也随之增加,因为需要额外的空间来记录未达到K次访问的数据,并且需要维护一个优先级队列。 2Q算法,又称为Two Queues,是LRU-2的一种变体,它将数据分为FIFO队列和...

    常用的缓存淘汰算法.pdf

    常见的缓存淘汰算法有LFU、LRU、ARC、FIFO、MRU等。 1. 最不经常使用算法(LFU):LFU缓存算法使用一个计数器来记录条目被访问的频率。通过使用LFU缓存算法,最低访问数的条目首先被移除。这个方法并不经常使用,...

    链表(上):如何实现LRU缓存淘汰算法.pdf

    ### 链表在LRU缓存淘汰算法中的应用 #### LRU缓存淘汰算法简介 LRU(Least Recently Used,最近最少使用)缓存淘汰算法是一种常用的缓存管理策略,用于解决缓存空间有限的问题。当缓存满了并且新的数据需要加入缓存...

    06丨链表(上):如何实现LRU缓存淘汰算法?1

    "链表数据结构与LRU缓存淘汰算法" 链表是一种基础的数据结构,它广泛应用于软件开发和硬件设计中。今天我们将讨论如何使用链表来实现LRU缓存淘汰算法。 首先,让我们来讨论缓存的概念。缓存是一种提高数据读取性能...

    缓存、缓存算法和缓存框架简介 - 文章 - 伯乐在线.pdf

    缓存算法,又称缓存淘汰算法,是指在缓存容量达到上限时,用于决定哪些数据应该被保留,哪些数据应该被替换出去的一系列算法。常见的缓存算法包括最近最少使用(LRU)、先进先出(FIFO)、时钟算法(Clock)等。这些...

    PyPI 官网下载 | lru-dict-1.1.6.tar.gz

    2. **高性能**:通过直接操作字典的内部结构,lru-dict实现了近乎原生字典的访问速度,同时带有缓存淘汰机制。 3. **简单易用**:它的API设计简洁,易于理解和集成到项目中,只需几行代码即可实现高效缓存。 4. **...

    06丨链表(上):如何实现LRU缓存淘汰算法?.pdf

    链表(上):如何实现 LRU 缓存淘汰算法? 链表是一种基础的数据结构,学习链表有什么用呢?为了回答这个问题,我们来讨论一个经典的链表应用场景,那就是 LRU 缓存淘汰算法。缓存是一种提高数据读取性能的技术,在...

    调度算法中的经典-LRU算法的实现(绝对值得收藏)

    LRU(Least Recently Used,最近最少使用)算法是一种常见的页面替换策略,广泛应用于缓存管理、内存调度等领域。它的核心思想是优先淘汰那些最长时间没有被访问过的数据,假设未来最不常使用的数据也将继续不常被...

    c语言实现的LRU算法

    LRU(Least Recently Used)算法是一种常用的页面替换策略,它基于“最近最少使用”的原则,即当内存空间不足时,最长时间未被访问过的页面优先被淘汰。在C语言中实现LRU算法,需要理解数据结构和算法的基础知识,...

    15 当Buffer Pool中的缓存页不够的时候,如何基于LRU算法淘汰部分缓存.pdf

    当Buffer Pool中的缓存页不够用时,必须淘汰一些缓存页来腾出空间,这时就需要用到缓存淘汰算法。这篇文章主要介绍了基于LRU(Least Recently Used,最近最少使用)算法在Buffer Pool中的缓存页淘汰机制。 首先,...

    python-LRU缓存.zip

    LRU缓存算法是一种常用的缓存淘汰策略,它根据数据项最近被使用的时间来决定哪些数据项应该从缓存中移除,以确保缓存中始终存放着最近最可能被使用的数据项。这个压缩文件可能包括LRU缓存的实现代码、测试样例、使用...

    前端开源库-lighter-lru-cache

    其中,LRU(Least Recently Used,最近最少使用)算法是一种常用的缓存淘汰策略。本文将详细介绍一个前端开源库——`lighter-lru-cache`,它是一个轻量级的JavaScript LRU缓存实现。 `lighter-lru-cache`库的核心...

    LRU算法LRU算法LRU算法LRU算法

    LRU(Least Recently Used...总结一下,LRU算法是一种基于历史访问行为的缓存策略,它通过淘汰最近最少使用的数据来优化内存使用。在实际应用中,如数据库缓存、操作系统的页面替换策略等,LRU算法都展现出良好的性能。

    东南大学操作系统实验——实现LRU算法及其近似算法

    时间复杂度方面,LRU算法通常表现为O(1),因为查找、插入和删除操作都在平均情况下具有常数时间复杂度。空间复杂度取决于缓存的最大大小,即`_max_size`。实现难度在于如何有效地维护页面的访问顺序和页面映射。 ...

    lru算法实验报告

    在实际应用中,LRU 算法常用于缓存管理和数据库系统,以优化数据的访问效率。例如,当内存不足以存放所有数据时,LRU 可以帮助决定哪些数据应该被暂时移除,以便为更重要的数据腾出空间。由于 LRU 的高效性和直观性...

    concurrentlinkedhashmap-lru-1.3.jar.zip

    首先,LRU是一种常用的缓存淘汰策略,它基于“最近最少使用”的原则,当缓存满时,优先淘汰最近最少使用的数据。在ConcurrentLinkedHashMap-LRU 1.3中,这种策略被巧妙地融入到并发环境下,保证了在多线程环境下的...

Global site tag (gtag.js) - Google Analytics