LRU(Least Recently Used)算法原理

2024-05-24 11:12

本文主要是介绍LRU(Least Recently Used)算法原理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

LRU(Least Recently Used)算法原理

一、简介

LRU(Least Recently Used)算法是一种常用的缓存淘汰策略,用于管理计算机系统中的缓存。当缓存满时,需要根据一定的策略淘汰掉一些数据,以便为新的数据腾出空间。LRU 算法的基本思想是:最近最少使用的数据最有可能在未来一段时间内不再被使用,因此应该优先淘汰这些数据。

二、原理概述

LRU 算法的核心是维护一个有序的数据结构,按照数据被访问的时间顺序排列。当缓存满时,移除最久未被访问的数据。LRU 算法可以通过多种数据结构实现,常见的有链表和哈希表结合的数据结构。

三、实现方法

3.1 链表实现

  • 使用一个双向链表存储缓存数据,链表的头部是最近访问的数据,尾部是最久未访问的数据。
  • 每次访问某个数据时,将该数据移动到链表头部。
  • 当缓存满时,移除链表尾部的数据。

3.2 链表实现哈希表+双向链表实现

  • 使用哈希表存储缓存中的数据,以便快速查找。
  • 使用双向链表维护访问顺序,链表的头部是最近访问的数据,尾部是最久未访问的数据。
  • 每次访问某个数据时,通过哈希表定位到链表中的节点,并将该节点移动到链表头部。
  • 当缓存满时,移除链表尾部的节点,并在哈希表中删除相应的条目。

四、示例代码

以下是使用哈希表和双向链表实现 LRU 缓存的示例代码:

import java.util.HashMap;class LRUCache {private class Node {int key, value;Node prev, next;Node(int key, int value) {this.key = key;this.value = value;}}private final int capacity;private HashMap<Integer, Node> map;private Node head, tail;public LRUCache(int capacity) {this.capacity = capacity;map = new HashMap<>();head = new Node(0, 0);tail = new Node(0, 0);head.next = tail;tail.prev = head;}public int get(int key) {if (map.containsKey(key)) {Node node = map.get(key);remove(node);insertToHead(node);return node.value;} else {return -1;}}public void put(int key, int value) {if (map.containsKey(key)) {Node node = map.get(key);remove(node);} else if (map.size() == capacity) {map.remove(tail.prev.key);remove(tail.prev);}Node node = new Node(key, value);insertToHead(node);map.put(key, node);}private void remove(Node node) {node.prev.next = node.next;node.next.prev = node.prev;}private void insertToHead(Node node) {node.next = head.next;node.prev = head;head.next.prev = node;head.next = node;}
}

4.1 类定义和成员变量

  • LRUCache 类定义了 LRU 缓存。
  • Node 内部类用于表示双向链表的节点,包含缓存的 key 和 value 以及指向前后节点的指针 prev 和 next。
  • capacity 是缓存的最大容量。
  • map 是哈希表,用于快速查找缓存中的节点。
  • head 和 tail 是双向链表的虚拟头尾节点,方便对链表进行操作。

4.2 构造函数

  • 初始化缓存的容量 capacity。
  • 初始化哈希表 map。
  • 初始化双向链表的虚拟头节点 head 和虚拟尾节点 tail,并将它们连接起来,形成一个空的双向链表。

4.3 获取缓存值

  • get 方法用于获取缓存中的值。
  • 如果 key 存在于哈希表中,获取对应的节点 node,调用 remove 方法将节点从链表中移除,并调用 insertToHead 方法将节点插入链表头部,表示最近使用,然后返回节点的 value。
  • 如果 key 不存在于哈希表中,返回 -1。

4.4 放入缓存值

  • put 方法用于在缓存中插入或更新一个值。
  • 如果 key 已存在于哈希表中,获取对应的节点 node,调用 remove 方法将节点从链表中移除。
  • 如果 key 不存在并且缓存已满,从哈希表中移除最久未使用的节点(即链表尾部节点 tail.prev),并调用 remove 方法将链表尾部节点移除。
  • 创建新的节点 node,调用 insertToHead 方法将新节点插入链表头部,并将新节点添加到哈希表 map 中。

4.5 移除节点

  • remove 方法用于从双向链表中移除一个节点 node。通过调整节点的前后指针,将 node 从链表中断开。

4.6 插入节点到链表头部

  • insertToHead 方法用于将一个节点 node 插入到双向链表的头部。通过调整指针,将 node 插入到虚拟头节点 head 和原第一个节点之间。

五、性能分析

  • 时间复杂度:LRU 缓存的 get 和 put 操作的时间复杂度都是 O(1),因为哈希表的查找和链表的插入/删除操作都是常数时间复杂度。
  • 空间复杂度:空间复杂度是 O(n),其中 n 是缓存的容量。哈希表和链表的空间开销与缓存的容量成线性关系。

六、优缺点

  • 优点
    • LRU 算法简单易实现。
    • 能够较好地淘汰不常使用的数据,提高缓存的命中率。
  • 缺点
    • 在高并发场景下,链表操作可能会成为性能瓶颈。
    • 维护链表的操作会带来一定的开销。

七、实际应用

LRU 算法广泛应用于各类缓存系统,如操作系统的页面置换、数据库的缓存机制、浏览器的缓存管理等。它有效地提高了系统的性能和资源利用率,是缓存淘汰策略中的经典算法之一。

总结

LRU 算法通过淘汰最近最少使用的数据,维护了缓存的高效性。通过哈希表和双向链表的结合,实现了 O(1) 时间复杂度的缓存操作。尽管在高并发场景下可能存在性能瓶颈,但 LRU 算法仍然是实际应用中最为常用和有效的缓存淘汰策略之一。

这篇关于LRU(Least Recently Used)算法原理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/998241

相关文章

MySQL中的MVCC底层原理解读

《MySQL中的MVCC底层原理解读》本文详细介绍了MySQL中的多版本并发控制(MVCC)机制,包括版本链、ReadView以及在不同事务隔离级别下MVCC的工作原理,通过一个具体的示例演示了在可重... 目录简介ReadView版本链演示过程总结简介MVCC(Multi-Version Concurr

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

Python中的随机森林算法与实战

《Python中的随机森林算法与实战》本文详细介绍了随机森林算法,包括其原理、实现步骤、分类和回归案例,并讨论了其优点和缺点,通过面向对象编程实现了一个简单的随机森林模型,并应用于鸢尾花分类和波士顿房... 目录1、随机森林算法概述2、随机森林的原理3、实现步骤4、分类案例:使用随机森林预测鸢尾花品种4.1

Redis主从复制的原理分析

《Redis主从复制的原理分析》Redis主从复制通过将数据镜像到多个从节点,实现高可用性和扩展性,主从复制包括初次全量同步和增量同步两个阶段,为优化复制性能,可以采用AOF持久化、调整复制超时时间、... 目录Redis主从复制的原理主从复制概述配置主从复制数据同步过程复制一致性与延迟故障转移机制监控与维

SpringCloud配置动态更新原理解析

《SpringCloud配置动态更新原理解析》在微服务架构的浩瀚星海中,服务配置的动态更新如同魔法一般,能够让应用在不重启的情况下,实时响应配置的变更,SpringCloud作为微服务架构中的佼佼者,... 目录一、SpringBoot、Cloud配置的读取二、SpringCloud配置动态刷新三、更新@R

Redis主从复制实现原理分析

《Redis主从复制实现原理分析》Redis主从复制通过Sync和CommandPropagate阶段实现数据同步,2.8版本后引入Psync指令,根据复制偏移量进行全量或部分同步,优化了数据传输效率... 目录Redis主DodMIK从复制实现原理实现原理Psync: 2.8版本后总结Redis主从复制实

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个