力扣 - 146. LRU缓存机制
题目
思路
- 利用双链表和HashMap来解题
- 看到链表题目,我们可以使用头尾结点可以更好进行链表操作和边界判断等
- 还需要使用size变量来存储双链表的当前长度
- 调用get,如果存在的话,我们可以调用将在双链表中的结点通过修改指针移动到第一个;如果调用put,我们先判断是否存在该结点,如果不存在,可直接将链接插入即可,同时size++,如果存在的话,先删除原来的结点,再将新结点push到头部即可
代码
class LRUCache {
class Node {
int key;
int val;
Node pre;
Node next;
public Node() {}
public Node(int key, int val) {
this.key = key;
this.val = val;
}
}
// 哈希表中的node和双链表的node是同一个结点
private HashMap<Integer, Node> cache;
private Node dummyHead;
private Node dummyTail;
private int size;
private int capacity;
public LRUCache(int capacity) {
cache = new HashMap<>();
dummyHead = new Node();
dummyTail = new Node();
dummyHead.next = dummyTail;
dummyTail.pre = dummyHead;
this.capacity = capacity;
this.size = 0;
}
public int get(int key) {
// 先获取看看结点存不存在
Node node = cache.get(key);
// 如果存在的话,将当前访问的结点移动到链表头,并且返回值
if (node != null) {
moveToHead(node);
return node.val;
}
// 不存在的话就返回-1
return -1;
}
public void put(int key, int value) {
// 也是先看看结点是否存在
Node node = cache.get(key);
// 如果存在,那么要做的操作就是将结点移动到链表头,然后更新结点值即可
if (node != null) {
node.val = value;
moveToHead(node);
} else {
// 如果不存在的话我们就要创建新结点插入
Node newNode = new Node(key, value);
// 先添加到哈希表中
cache.put(key, newNode);
// 再添加到链表中
addToHead(newNode);
// 并且长度+1
size++;
// 因为我们设定最大容量,我们还要判断新put的结点后,容量是否超过了capacity,超过了话,删除最后一个结点,并且长度-1
if (size > capacity) {
Node tail = removeTail();
cache.remove(tail.key);
size--;
}
}
}
/**
* 将node移动到最前面
*/
private void moveToHead(Node node) {
if (size > 0 && size <= capacity) {
remove(node);
addToHead(node);
}
}
/**
* 删除node结点
*/
private void remove(Node node) {
if (size > 0) {
node.pre.next = node.next;
node.next.pre = node.pre;
}
}
/**
* 删除最后一个结点
*/
private Node removeTail() {
if (size > 0) {
Node node = dummyTail.pre;
remove(node);
return node;
}
return null;
}
/**
* 添加新结点到第一位去
*/
private void addToHead(Node node) {
node.next = dummyHead.next;
dummyHead.next = node;
node.next.pre = node;
node.pre = dummyHead;
}
}
复杂度分析
- 时间复杂度:\(O(1)\)
- 空间复杂度:\(O(N)\),其中 N 为初始化的 capacity 容量
力扣 - 146. LRU缓存机制的更多相关文章
- 【golang必备算法】 Letecode 146. LRU 缓存机制
力扣链接:146. LRU 缓存机制 思路:哈希表 + 双向链表 为什么必须要用双向链表? 因为我们需要删除操作.删除一个节点不光要得到该节点本身的指针,也需要操作其前驱节点的指针,而双向链表才能支持 ...
- Java实现 LeetCode 146 LRU缓存机制
146. LRU缓存机制 运用你所掌握的数据结构,设计和实现一个 LRU (最近最少使用) 缓存机制.它应该支持以下操作: 获取数据 get 和 写入数据 put . 获取数据 get(key) - ...
- 146. LRU 缓存机制 + 哈希表 + 自定义双向链表
146. LRU 缓存机制 LeetCode-146 题目描述 题解分析 java代码 package com.walegarrett.interview; /** * @Author WaleGar ...
- 【力扣】146. LRU缓存机制
运用你所掌握的数据结构,设计和实现一个 LRU (最近最少使用) 缓存机制.它应该支持以下操作: 获取数据 get 和 写入数据 put . 获取数据 get(key) - 如果关键字 (key) ...
- [Leetcode]146.LRU缓存机制
Leetcode难题,题目为: 运用你所掌握的数据结构,设计和实现一个 LRU (最近最少使用) 缓存机制.它应该支持以下操作: 获取数据 get 和 写入数据 put . 获取数据 get(key ...
- 146. LRU缓存机制
题目描述 运用你所掌握的数据结构,设计和实现一个LRU (最近最少使用) 缓存机制.它应该支持以下操作: 获取数据 get 和 写入数据 put . 获取数据 get(key) - 如果密钥 (key ...
- leetcode:146. LRU缓存机制
题目描述: 运用你所掌握的数据结构,设计和实现一个 LRU (最近最少使用) 缓存机制.它应该支持以下操作: 获取数据 get 和 写入数据 put . 获取数据 get(key) - 如果密钥 ( ...
- LeetCode 146. LRU缓存机制(LRU Cache)
题目描述 运用你所掌握的数据结构,设计和实现一个 LRU (最近最少使用) 缓存机制.它应该支持以下操作: 获取数据 get 和 写入数据 put . 获取数据 get(key) - 如果密钥 (k ...
- Leetcode 146. LRU 缓存机制
前言 缓存是一种提高数据读取性能的技术,在计算机中cpu和主内存之间读取数据存在差异,CPU和主内存之间有CPU缓存,而且在内存和硬盘有内存缓存.当主存容量远大于CPU缓存,或磁盘容量远大于主存时,哪 ...
随机推荐
- Nuxt/Vue自定义导航栏Topbar+标签栏Tabbar组件
基于Vue.js实现自定义Topbar+Tabbar组件|仿咸鱼底部凸起导航 最近一直在倒腾Nuxt项目,由于Nuxt.js是基于Vue.js的服务端渲染框架,只要是会vue,基本能很快上手了. 一般 ...
- Linux批量查找与替换
Linux批量查找并替换文件夹下所有文件的内容 经常要使用到 Linux的批量查找与替换,这里为大家介绍使用 sed 命令和 grep 命令的结合来实现查找文件中的内容并替换. 语法格式: sed - ...
- TP5上传图片到七牛云,并且删除七牛云的图片
一,通过composer 下载七牛云 sdk composer require qiniu/php-sdk 二,手动下载七牛云sdk 1,https://developer.qiniu.com/kod ...
- volatile、ThreadLocal的使用场景和原理
并发编程中的三个概念 原子性 一个或多个操作.要么全部执行完成并且执行过程不会被打断,要么不执行.最常见的例子:i++/i--操作.不是原子性操作,如果不做好同步性就容易造成线程安全问题. 可见性 多 ...
- 基于python实现链式队列代码
""" 链式存储-队列 linkqueue.py 代码实现 思路: 1.入队, 2.出队, 3.判断空满 """ # 异常类 class Q ...
- 什么是 C 和 C ++ 标准库?学编程的你应该知道这些知识!
简要介绍编写C/C ++应用程序的领域,标准库的作用以及它是如何在各种操作系统中实现的. 我已经接触C++一段时间了,一开始就让我感到疑惑的是其内部结构:我所使用的内核函数和类从何而来? 谁发明了它们 ...
- logstash-安装
1.下载 cd /usr/local/src wget https://mirrors.huaweicloud.com/logstash/7.8.0/logstash-7.8.0.tar.gz ...
- 第三十二章 Linux常规练习题(一)
一.练习题一 1.超级用户(管理员用户)提示符是____,普通用户提示符是____.2.linux关机重启的命令有哪些 ?3.bash是什么?4.bash特性, 常见的bash特性有哪些?5.网卡的配 ...
- Django ORM Queryset 的缓存机制, 惰性查询简述
在Django的ORM中 必须注意由于QuerySet的 cache导致的数据获取不正确的问题 在哪些情况下不会出发QuerySet缓存? 隐式存储QuerySet(查询语句没有显示赋值给变量而直接进 ...
- Codeforces Educational Round 92 赛后解题报告(A-G)
Codeforces Educational Round 92 赛后解题报告 惨 huayucaiji 惨 A. LCM Problem 赛前:A题嘛,总归简单的咯 赛后:A题这种**题居然想了20m ...