[Leedcode][JAVA][第146题][LRU][哈希表][双向链表]

【问题描述】

LFU
运用你所掌握的数据结构,设计和实现一个  LRU (最近最少使用) 缓存机制。它应该支持以下操作: 获取数据 get 和 写入数据 put 。获取数据 get(key) - 如果密钥 (key) 存在于缓存中,则获取密钥的值(总是正数),否则返回 -1。
写入数据 put(key, value) - 如果密钥不存在,则写入其数据值。当缓存容量达到上限时,它应该在写入新数据之前删除最久未使用的数据值,从而为新的数据值留出空间。进阶:你是否可以在 O(1) 时间复杂度内完成这两种操作?示例:LRUCache cache = new LRUCache( 2 /* 缓存容量 */ );cache.put(1, 1);
cache.put(2, 2);
cache.get(1);       // 返回  1
cache.put(3, 3);    // 该操作会使得密钥 2 作废
cache.get(2);       // 返回 -1 (未找到)
cache.put(4, 4);    // 该操作会使得密钥 1 作废
cache.get(1);       // 返回 -1 (未找到)
cache.get(3);       // 返回  3
cache.get(4);       // 返回  4

【解答思路】

1. 缓存是有限的,在缓存满的时候,删除哪些元素,就有不同的缓存删除策略。

LRU (Least Recently Used)缓存机制

  • 在缓存满的时候,删除缓存里最久未使用的数据,然后再放入新元素;
  • 数据的访问时间很重要,访问时间距离现在最近,就最不容易被删除。
    image.png

时间复杂度:O(1) 空间复杂度:O(N)

import java.util.Map;public class LRUCache {private Map<Integer, ListNode> map;/*** 双链表结点类*/private class ListNode {private Integer key;private Integer value;/*** 前驱结点 precursor*/private ListNode pre;/*** 后继结点 successor(写成 next 是照顾单链表的表示习惯)*/private ListNode next;public ListNode() {}public ListNode(Integer key, Integer value) {this.key = key;this.value = value;}}private int capacity;/*** 虚拟头结点没有前驱*/private ListNode dummyHead;/*** 虚拟尾结点没有后继*/private ListNode dummyTail;public LRUCache(int capacity) {map = new HashMap<>(capacity);this.capacity = capacity;dummyHead = new ListNode(-1, -1);dummyTail = new ListNode(-1, -1);// 初始化链表为 head <-> taildummyHead.next = dummyTail;dummyTail.pre = dummyHead;}/*** 如果存在,把当前结点移动到双向链表的头部** @param key* @return*/public int get(int key) {if (map.containsKey(key)) {ListNode node = map.get(key);int val = node.value;// 把当前 node 移动到双向链表的头部moveNode2Head(key);return val;} else {return -1;}}/*** 如果哈希表的容量满了,就要删除一个链表末尾元素,然后在链表头部插入新元素** @param key* @param value*/public void put(int key, int value) {if (map.containsKey(key)) {// 1、更新 valuemap.get(key).value = value;// 2、把当前 node 移动到双向链表的头部moveNode2Head(key);return;}// 放元素的操作是一样的if (map.size() == capacity) {// 如果满了ListNode oldTail = removeTail();// 设计 key 就是为了在这里删除map.remove(oldTail.key);}// 然后添加元素ListNode newNode = new ListNode(key, value);map.put(key, newNode);addNode2Head(newNode);}// 为了突出主干逻辑,下面是 3 个公用的方法/*** 删除双链表尾部结点*/private ListNode removeTail() {ListNode oldTail = dummyTail.pre;ListNode newTail = oldTail.pre;// 两侧结点建立连接newTail.next = dummyTail;dummyTail.pre = newTail;// 释放引用oldTail.pre = null;oldTail.next = null;return oldTail;}/*** 把当前 key 指向的结点移到双向链表的头部** @param key*/private void moveNode2Head(int key) {// 1、先把 node 拿出来ListNode node = map.get(key);// 2、原来 node 的前驱和后继接上node.pre.next = node.next;node.next.pre = node.pre;// 3、再把 node 放在末尾addNode2Head(node);}/*** 在双链表的头部新增一个结点** @param newNode*/private void addNode2Head(ListNode newNode) {// 1、当前头结点ListNode oldHead = dummyHead.next;// 2、末尾结点的后继指向新结点oldHead.pre = newNode;// 3、设置新结点的前驱和后继newNode.pre = dummyHead;newNode.next = oldHead;// 4、更改虚拟头结点的后继结点dummyHead.next = newNode;}public static void main(String[] args) {LRUCache cache = new LRUCache(2);cache.put(1, 1);cache.put(2, 2);System.out.println(cache.map.keySet());int res1 = cache.get(1);System.out.println(res1);cache.put(3, 3);int res2 = cache.get(2);System.out.println(res2);int res3 = cache.get(3);System.out.println(res3);cache.put(4, 4);System.out.println(cache.map.keySet());int res4 = cache.get(1);System.out.println(res4);int res5 = cache.get(3);System.out.println(res5);int res6 = cache.get(4);System.out.println(res6);}
}作者:liweiwei1419
链接:https://leetcode-cn.com/problems/lru-cache/solution/ha-xi-biao-shuang-xiang-lian-biao-java-by-liweiw-2/

【总结】

1.缓存删除机制

  • LRU (Least Recently Used)缓存机制[第146题]
  • LFU (Least Frequently Used,最不经常使用)缓存机制[第460
    题]
  1. 时间空间tradeoff
    存取数据时间性能最好 - 哈希表
    访问某个数据,时间优先级得提前,删除末尾结点的需求,头尾访问数据最快 - 双向链表

  2. LinkedList 双向链表 本题需要定制化操作

参考链接:https://leetcode-cn.com/problems/lru-cache/solution/ha-xi-biao-shuang-xiang-lian-biao-java-by-liweiw-2/

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/426006.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

wdcp 升级 php7,SugarHosts糖果主机升级硬盘流量 且PHP默认版本升级至PHP7.2

我们对于糖果主机SugarHosts商家应该还算是比较熟悉的&#xff0c;早期老左还没接触云服务器的时候建站都是用的虚拟主机&#xff0c;那时候我们知道有Bluehost、Justhost、Hostmonster等&#xff0c;当然还有糖果主机那时候也算是比较流行的主机商其主要是支持简体中文和支付宝…

前后端不分离与分离

前后端不分离 在前后端不分离的应用模式中&#xff0c;前端页面看到的效果是由后端控制&#xff0c;由后端渲染页面或重定向&#xff0c;后端需要控制前端的展示&#xff0c;前端与后端的耦合度很&#xff0c;这种应用模式比较适合纯网页应用&#xff0c;但是当后端对接APP时&a…

java学习(155):序列化

import javax.imageio.IIOException; import java.io.*; import java.util.Date;//字符输入流 public class FileManagerChar {public static void readCharFile(File file){FileReader fileReadernull;//文本输入流if(file.exists()){try {fileReader new FileReader( file )…

matlab2008报错,simulink的.mdl的文件在matlab2008a中打开报错,高手指点下

哈哈&#xff0c;不同版本字体的问题。运行下边的命令&#xff1a;bdclose all; set_param(0,CharacterEncoding, ISO-8859-1)如果不行再换一下字体。可以参照一下命令的介绍&#xff0c;能够使用联机帮助&#xff0c;这是很关键的。另外&#xff0c;不同的版本还会出现模型问题…

[Leedcode][JAVA][第445题][链表][栈]

【问题描述】445. 两数相加 II 给你两个 非空 链表来代表两个非负整数。数字最高位位于链表开始位置。它们的每个节点只存储一位数字。将这两数相加会返回一个新的链表。你可以假设除了数字 0 之外&#xff0c;这两个数字都不会以零开头。进阶&#xff1a;如果输入链表不能修改…

Leetcode143. Reorder List重排链表

给定一个单链表 L&#xff1a;L0→L1→…→Ln-1→Ln &#xff0c; 将其重新排列后变为&#xff1a; L0→Ln→L1→Ln-1→L2→Ln-2→… 你不能只是单纯的改变节点内部的值&#xff0c;而是需要实际的进行节点交换。 示例 1: 给定链表 1->2->3->4, 重新排列为 1->4-&g…

java学习(156):反序列化

import javax.imageio.IIOException; import java.io.*; import java.util.Date;//字符输入流 public class FileManagerChar {public static void readCharFile(File file){FileReader fileReadernull;//文本输入流if(file.exists()){try {fileReader new FileReader( file )…

[Leedcode][JAVA][第355题][设计推特][面向对象][哈希表][链表][优先队列]

【问题描述】355 设计推特 设计一个简化版的推特(Twitter)&#xff0c;可以让用户实现发送推文&#xff0c;关注/取消关注其他用户&#xff0c;能够看见关注人&#xff08;包括自己&#xff09;的最近十条推文。你的设计需要支持以下的几个功能&#xff1a;postTweet(userId, …

java中sofa并发访问,云上的日子:用块存储、文件存储还是对象存储?

当今最流行的存储类型是哪种?答案毫无疑问是&#xff1a;块存储、文件存储还有对象存储!作为伴随着云计算兴起的存储类型&#xff0c;这三种存储绝对是这个时代的存储明星&#xff0c;用C位出道来形容再合适不过。那么&#xff0c;在云上的日子&#xff0c;究竟该用块存储、文…

java学习(157):线程的引入

//线程 public class test102 extends Thread{public void run(){while (true){System.out.println("我是线程1"this.getName());try {Thread.sleep( 1000 );}catch (InterruptedException e){e.printStackTrace();}}} }//线程 public class test103 extends Thread{…

【读书笔记】计算机是怎样跑起来的

2018年底读的&#xff0c;才整理完笔记。说说感想吧&#xff0c;每个章节虽然都是基础的知识&#xff0c;但是每次读起来还是收获颇多&#xff0c;很多自己说不清楚或者不理解的概念&#xff0c;感觉在认识层面有些结成网了。对于每个方面的专业知识&#xff0c;要按需深入学习…

[剑指offer][JAVA]面试题[第23题][合并K个排序链表][分治][优先队列]

【问题描述】23.合并K个排序链表 合并 k 个排序链表&#xff0c;返回合并后的排序链表。请分析和描述算法的复杂度。 示例: 输入: [1->4->5,1->3->4,2->6 ] 输出: 1->1->2->3->4->4->5->6【解答思路】 1. 两两合并 public class Soluti…

源码安装的php如何启动脚本,PHP源码编译安装管理常用脚本

#!/bin/sh# 编译安装管理PHPAppphpAppNamePHPAppBase/AppAppDir$AppBase/$AppAppProg$AppDir/sbin/php-fpmAppIni$AppDir/etc/php.iniAppConf$AppDir/etc/php-fpm.confExtensionDir$($AppDir/bin/php-config --extension-dir)AppSrcBase/App/srcAppSrcFile$App-*.tar.*AppSrcDi…

java学习(158):线程的常用方法yield,优先级

//线程 public class test102 extends Thread{public void run(){while (true){System.out.println("我是线程1"this.getName());try {Thread.sleep( 1000 );}catch (InterruptedException e){e.printStackTrace();}}} }//线程 public class test103 extends Thread{…

[Leedcode][JAVA][第542题][01矩阵][BFS]

【问题描述】542. 01 矩阵 给定一个由 0 和 1 组成的矩阵&#xff0c;找出每个元素到最近的 0 的距离。 两个相邻元素间的距离为 1 。 示例 : 输入: 0 0 0 0 1 0 1 1 1 输出: 0 0 0 0 1 0 1 2 1 注意: 给定矩阵的元素个数不超过 10000。 给定矩阵中至少有一个元素是 0。 矩阵…

php where or,php – laravel中whereOr和orWhere之间的区别

我在Laravel的代码中使用whereOr和orWhere都可以,但有时会产生不同的结果$user_query User::select( users.id, users.username,users.first_name,users.last_name, users.photo )->where(users.status,1)->where(users.id,!,$id)->where(users.username,like,%.$sea…

java学习(159):两个线程共同完成1到100计算

//利用两个线程实现1到100的计算 public class MyRannable implements java.lang.Runnable{private Thread th_0;private Thread th_2;int sum0;//存储累加和的结果int i1;public void run(){String thNameThread.currentThread().getName();//获取当前线程的名字while (i<…

约瑟夫问题(Josephus Problem)的两种快速递归算法

参考&#xff1a;http://haoyuanliu.github.io/2016/04/18/Josephus/ 转载于:https://www.cnblogs.com/xiaoshayu520ly/p/10263113.html

java学习(160):interrupt方法

import java.util.PrimitiveIterator;public class Classroon implements Runnable {private Thread student;//学生线程private Thread teacher;//老师线程public Classroon(){student new Thread( this,"小迷糊" );teacher new Thread( this,"大教授" …

elementui图片上传php,vue+element-ui+富文本————图片上传

最近用vueelement-ui开发一套后台管理系统&#xff0c;其中项目中需要用到富文本编辑器&#xff0c;这里总结下开发过程中遇到的坑和使用方法。刚开始用的是vue-quill-editor结合element-ui上传图片到服务器&#xff0c;name问题来了 按照官方文档上面的方式。下面是我的代码cl…