本期封面作者:A17

—————  两个月前  —————

用户信息当然是存在数据库里。但是由于我们对用户系统的性能要求比较高,显然不能每一次请求都去查询数据库。

所以,小灰在内存中创建了一个哈希表作为缓存,每次查找一个用户的时候先在哈希表中查询,以此提高访问性能。

很快,用户系统上线了,小灰美美地休息了几天。

一个多月之后......

———————————————

什么是哈希链表呢?

我们都知道,哈希表是由若干个Key-Value所组成。在“逻辑”上,这些Key-Value是无所谓排列顺序的,谁先谁后都一样。

在哈希链表当中,这些Key-Value不再是彼此无关的存在,而是被一个链条串了起来。每一个Key-Value都具有它的前驱Key-Value、后继Key-Value,就像双向链表中的节点一样。

这样一来,原本无序的哈希表拥有了固定的排列顺序。

让我们以用户信息的需求为例,来演示一下LRU算法的基本思路:

1.假设我们使用哈希链表来缓存用户信息,目前缓存了4个用户,这4个用户是按照时间顺序依次从链表右端插入的。

2.此时,业务方访问用户5,由于哈希链表中没有用户5的数据,我们从数据库中读取出来,插入到缓存当中。这时候,链表中最右端是最新访问到的用户5,最左端是最近最少访问的用户1。

3.接下来,业务方访问用户2,哈希链表中存在用户2的数据,我们怎么做呢?我们把用户2从它的前驱节点和后继节点之间移除,重新插入到链表最右端。这时候,链表中最右端变成了最新访问到的用户2,最左端仍然是最近最少访问的用户1。

4.接下来,业务方请求修改用户4的信息。同样道理,我们把用户4从原来的位置移动到链表最右侧,并把用户信息的值更新。这时候,链表中最右端是最新访问到的用户4,最左端仍然是最近最少访问的用户1。

5.后来业务方换口味了,访问用户6,用户6在缓存里没有,需要插入到哈希链表。假设这时候缓存容量已经达到上限,必须先删除最近最少访问的数据,那么位于哈希链表最左端的用户1就会被删除掉,然后再把用户6插入到最右端。

以上,就是LRU算法的基本思路。

  1. private Node head;

  2. private Node end;

  3. //缓存存储上限

  4. private int limit;

  5. private HashMap<String, Node> hashMap;

  6. public LRUCache(int limit) {

  7.    this.limit = limit;

  8.    hashMap = new HashMap<String, Node>();

  9. }

  10. public String get(String key) {

  11.    Node node = hashMap.get(key);

  12.    if (node == null){

  13.        return null;

  14.    }

  15.    refreshNode(node);

  16.    return node.value;

  17. }

  18. public void put(String key, String value) {

  19.    Node node = hashMap.get(key);

  20.    if (node == null) {

  21.        //如果key不存在,插入key-value

  22.        if (hashMap.size() >= limit) {

  23.            String oldKey = removeNode(head);

  24.            hashMap.remove(oldKey);

  25.        }

  26.        node = new Node(key, value);

  27.        addNode(node);

  28.        hashMap.put(key, node);

  29.    }else {

  30.        //如果key存在,刷新key-value

  31.        node.value = value;

  32.        refreshNode(node);

  33.    }

  34. }

  35. public void remove(String key) {

  36.    Node node = hashMap.get(key);

  37.    removeNode(node);

  38.    hashMap.remove(key);

  39. }

  40. /**

  41. * 刷新被访问的节点位置

  42. * @param  node 被访问的节点

  43. */

  44. private void refreshNode(Node node) {

  45.    //如果访问的是尾节点,无需移动节点

  46.    if (node == end) {

  47.        return;

  48.    }

  49.    //移除节点

  50.    removeNode(node);

  51.    //重新插入节点

  52.    addNode(node);

  53. }

  54. /**

  55. * 删除节点

  56. * @param  node 要删除的节点

  57. */

  58. private String removeNode(Node node) {

  59.    if (node == end) {

  60.        //移除尾节点

  61.        end = end.pre;

  62.    }else if(node == head){

  63.        //移除头节点

  64.        head = head.next;

  65.    } else {

  66.        //移除中间节点

  67.        node.pre.next = node.next;

  68.        node.next.pre = node.pre;

  69.    }

  70.    return node.key;

  71. }

  72. /**

  73. * 尾部插入节点

  74. * @param  node 要插入的节点

  75. */

  76. private void addNode(Node node) {

  77.    if(end != null) {

  78.        end.next = node;

  79.        node.pre = end;

  80.        node.next = null;

  81.    }

  82.    end = node;

  83.    if(head == null){

  84.        head = node;

  85.    }

  86. }

  87. class Node {

  88.    Node(String key, String value){

  89.        this.key = key;

  90.        this.value = value;

  91.    }

  92.    public Node pre;

  93.    public Node next;

  94.    public String key;

  95.    public String value;

  96. }

  97. public static void main(String[] args) {

  98.    LRUCache lruCache = new LRUCache(5);

  99.    lruCache.put("001", "用户1信息");

  100.    lruCache.put("002", "用户1信息");

  101.    lruCache.put("003", "用户1信息");

  102.    lruCache.put("004", "用户1信息");

  103.    lruCache.put("005", "用户1信息");

  104.    lruCache.get("002");

  105.    lruCache.put("004", "用户2信息更新");

  106.    lruCache.put("006", "用户6信息");

  107.    System.out.println(lruCache.get("001"));

  108.    System.out.println(lruCache.get("006"));

  109. }

需要注意的是,这段不是线程安全的,要想做到线程安全,需要加上synchronized修饰符。

告诉大家一个好消息,小灰的《漫画算法》全面上架啦,在短短的两周里,本书一度霸占着各大畅销榜榜首!

扫码或者点击封面查看详情

小灰把两年多以来积累的漫画作品进行了筛选和优化,并加上了一些更为基础和系统的入门章节,最终完成了本书的六大篇章:

第一章 算法概述

介绍了算法和数据结构的相关概念,告诉大家算法是什么,数据结构又是什么,它们有哪些用途,如何分析时间复杂度,如何分析空间复杂度。

第二章 数据结构基础

介绍了最基本的数据结构,包括数组、链表、栈、队列、哈希表的概念和读写操作。

第三章 树

介绍了树和二叉树的概念、二叉树的各种遍历方式、二叉堆和优先队列的应用。

第四章 排序算法

介绍了几种典型的排序算法,包括冒泡排序、快速排序、堆排序、计数排序、桶排序。

第五章 面试中的算法

介绍了10余道职场上流行的算法面试题及详细的解题思路。例如怎样判断链表有环、怎样计算大整数相加等。

第六章 算法的实际应用

介绍了算法在职场上的一些应用,例如使用LRU算法来淘汰冷数据,使用Bitmap算法来统计用户特征等。

本书是全彩印制,书中的每一章、每一节、每一句话、每一幅图、每一行代码,都经过了小灰和编辑们的精心打磨,力求用最为直白的方式把知识讲明白、讲透彻。

早期的漫画中存在一些叙述错误和表达不清晰的地方,小灰在本书中做了修正和补充;同时书中增加了许多全新的篇章,使得本书的内容更加系统和全面。

对于渴望学习算法的小伙伴,无论你是正在学习计算机专业的学生,还是已经进入职场的新人,亦或是拥有多年工作经验却不擅长算法的老手,这本《漫画算法》都能帮助你告别对算法的恐惧,认识算法、掌握算法。

扫码或点击阅读原文购买

新品8折优惠中

漫画:什么是LRU算法?相关推荐

  1. LRU算法 -- 链表 完整实现

    LRU算法(Least Recently Used) 算是我们经常遇到的一种淘汰算法,其中内存管理模块进行内存页回收时有用到,针对不经常使用的内存页,LRU淘汰策略能够将该内存页回收给操作系统. 属于 ...

  2. 【腾讯三面】你能现场写一下LRU算法吗?

    " 金三银四,又到了换工作的黄金期.各位小伙伴们都准备好了吗? " 这句话大家是不是最近已经要看吐了呢? 每当这个时候,就证明招聘旺季又来啦~ 春招.校招.社招-- 那你真的准备好 ...

  3. LRU算法确定最后使用时间的顺序

    LRU算法是经常采用的页面置换算法,并被认为是相当好的,但是存在如何实现它的问题.LRU算法需要实际硬件的支持.其问题是怎么确定最后使用时间的顺序,对此有两种可行的办法: 1.计数器. 2.栈.

  4. java 最少使用(lru)置换算法_「面试」LRU了解么?看看LinkedHashMap如何实现LRU算法...

    以下内容均是本人原创,希望你看完之后能有更多更深入的了解,欢迎关注➕ 问题:使用Java完成一个简单的LRU算法 什么是LRU算法 LRU(Least Recently Used),也就是最近最少使用 ...

  5. Android图片缓存之Lru算法

    前言: 上篇我们总结了Bitmap的处理,同时对比了各种处理的效率以及对内存占用大小.我们得知一个应用如果使用大量图片就会导致OOM(out of memory),那该如何处理才能近可能的降低oom发 ...

  6. 使用java.util.LinkedList模拟实现内存页面置换算法--LRU算法

    一,LRU算法介绍 LRU是内存分配中"离散分配方式"之分页存储管理方式中用到的一个算法.每个进程都有自己的页表,进程只将自己的一部分页面加载到内存的物理块中,当进程在运行过程中, ...

  7. 2.LRU算法实现 [C++]

    LRU算法实现原理: 最近最久未使用页面置换算法(LRU) 当需要淘汰某一页时,选择在最近一段时间里最久没有被使用过的页淘汰. 其基本原理为:如果某一个页面被访问了,它很可能还要被访问:相反,如果它长 ...

  8. 操作系统之存储管理——FIFO算法和LRU算法

    操作系统之进程调度--优先权法和轮转法(附上样例讲解) 操作系统之银行家算法-详解流程及案例数据 操作系统之多线程编程-读者优先/写者优先详解 操作系统之存储管理--FIFO算法和LRU算法 操作系统 ...

  9. 编写程序来模拟计算机LRU算法,操作系统-页式虚拟存储管理程序模拟.doc

    实 验3: 页式虚拟存储管理程序模拟 实验目的: 编写程序来模拟计算机的两种调度方式: (1)先进先出算法 (2)最近最少使用算法 程序设计 FIFO() FIFO() LRU() Msize加1 S ...

最新文章

  1. spark集群使用hanlp进行分布式分词操作说明
  2. android 中使用TabHost控件实现微信界面的底部菜单效果
  3. python画椭圆-Python易学就会(五)turtle绘制椭圆与递归
  4. Mysql数据库主从及主主复制配置演示
  5. 【Python基础】pandas的骚操作:一行 pandas 代码搞定 Excel “条件格式”!
  6. 百度代码规范 -- PHP
  7. 统一设置网站html文件输出编码为utf-8,以解决乱码问题
  8. 本地网页服务器 跨域,nodejs搭建本地服务器轻松解决跨域问题
  9. npm使用taobao镜像
  10. 腾讯开源最大规模多标签图像数据集,刷新行业数据集基准
  11. [NOIP2015]金币
  12. JAVA中自增自减运算符(i++与++i的区别)
  13. 另类方法激活你的Winodws 2008
  14. python的ols_Python Statsmodels 统计包之 OLS 回归
  15. vss 迁入后,服务器上面的文件没有变化,VSS迁移
  16. serverTimezone
  17. C++动态壁纸软件的开发(含源文件)
  18. MYSQL 列转行方法
  19. 《RAFT-Stereo:Multilevel Recurrent Field Transforms for Stereo Matching》论文笔记
  20. 摄像头集中监控软件系统

热门文章

  1. 2015大型互联网公司校招都开始了,薪资你准备好了嘛?
  2. Quintum 语音网关设置方法
  3. 再谈javascript图片预加载经典技术
  4. 能说明你的Javascript技术很烂的五个原因
  5. 2020大部分知名公司Java面试题(一)
  6. linux+用户的shell,Linux用户管理(十)Linux Shell高级
  7. java中引用类型_您真的了解Java中的4种引用类型吗?
  8. 洛谷P3254 圆桌问题(最大流)
  9. LESS 的 operation 是 特性
  10. 最近在做托盘时,发现 CnTrayIcon1的OnClick 事件,不能被其它按钮来执行,蛋疼。...