思路
具体的思路在LeetCode官方解答中有,里面还有动画和视频,推荐看不懂下面思路的可以去看一下。
-
LRU 缓存机制可以通过哈希表辅以双向链表实现,我们用一个哈希表和一个双向链表维护所有在缓存中的键值对。 -
双向链表按照被使用的顺序存储了这些键值对,靠近头部的键值对是最近使用的,而靠近尾部的键值对是最久未使用的。 -
哈希表即为普通的哈希映射(HashMap ),通过缓存数据的键映射到其在双向链表中的位置。 这样以来,我们首先使用哈希表进行定位,找出缓存项在双向链表中的位置,随后将其移动到双向链表的头部,即可在 O(1)O(1) 的时间内完成 get 或者 put 操作。具体的方法如下:
- 对于
get 操作,首先判断key 是否存在: - 如果
key 不存在,则返回 -1; - 如果
key 存在,则 key 对应的节点是最近被使用的节点。通过哈希表定位到该节点在双向链表中的位置,并将其移动到双向链表的头部,最后返回该节点的值。 - 对于
put 操作,首先判断key 是否存在:
如果 key 不存在,使用 key 和 value 创建一个新的节点,在双向链表的头部添加该节点,并将 key 和该节点添加进哈希表中。然后判断双向链表的节点数是否超出容量,如果超出容量,则删除双向链表的尾部节点,并删除哈希表中对应的项;
如果 key 存在,则与 get 操作类似,先通过哈希表定位,再将对应的节点的值更新为 value ,并将该节点移到双向链表的头部。
- 上述各项操作中,访问哈希表的时间复杂度为
O(1)O(1) ,在双向链表的头部添加节点、在双向链表的尾部删除节点的复杂度也为 O(1)O(1) 。而将一个节点移到双向链表的头部,可以分成「删除该节点」和「在双向链表的头部添加节点」两步操作,都可以在 O(1)O(1) 时间内完成。
注意:在双向链表的实现中,可以使用一个伪头部(dummy head)和伪尾部(dummy tail)标记界限,这样在添加节点和删除节点的时候就不需要检查相邻的节点是否存在
不使用内置的LinkedHashMap ,使用哈希表+双向链表
代码
class LRUCache {
class Node{
public int key,val;
public Node next,prev;
public Node(int k ,int v){
this.key = k;
this.val = v;
}
}
class DoubleList{
private Node head,tail;
private int size;
public DoubleList(){
head = new Node(0,0);
tail = new Node(0,0);
head.next = tail;
tail.prev = head;
size = 0;
}
public void addLast(Node x){
x.prev = tail.prev;
x.next = tail;
tail.prev.next = x;
tail.prev = x;
size++;
}
public void remove (Node x){
x.prev.next = x.next;
x.next.prev = x.prev;
size--;
}
public Node removeFirst(){
if(head.next==tail) return null;
Node fist = head.next;
remove(fist);
return fist;
}
public int size() {
return size;
}
}
private HashMap<Integer,Node> map;
private DoubleList cache;
private int cap;
public LRUCache(int capacity) {
this.cap = capacity;
map = new HashMap<>();
cache = new DoubleList();
}
private void makeRecently (int key){
Node x = map.get(key);
cache.remove(x);
cache.addLast(x);
}
private void addRecently(int key,int val){
Node x = new Node(key,val);
cache.addLast(x);
map.put(key,x);
}
private void deleteKey(int key){
Node x = map.get(key);
cache.remove(x);
map.remove(key);
}
private void deleteLastRecently(){
Node x = cache.removeFirst();
int key = x.key;
map.remove(key);
}
public int get(int key) {
if(!map.containsKey(key)){
return -1;
}
makeRecently(key);
return map.get(key).val;
}
public void put(int key, int val) {
if(map.containsKey(key)){
deleteKey(key);
addRecently(key,val);
return;
}
if(cache.size()==this.cap){
deleteLastRecently();
}
addRecently(key,val);
}
}
说明
- 为什么不使用单链表而要使用双向链表?
在代码中就可以看出,我们需要进行删除操作(remove ),删除一个节点不仅要得到该节点本身的指针,也需要操作其前驱节点的指针,而只有双向链表才支持直接找前驱,这样才能保证时间复杂度为0(1),而单链表则达不到这样的效果。
- 为什么哈希表存储了
key ,为什么链表中还要存key 和val ,直接存val不行吗?
因为在deleteLastRecently 函数中,既需要删除节点,也需要得到key 。也就是当存储容量满了之后,不仅要删除最后一个Node 节点,还要把map ·中映射到该节点的key删除,而这个key 只能通过Node 节点获得。如果Node 结构中只存储了val ,那么就无法知道key 是什么了,也就无法通过key 去删除map 中的键了。
使用内置的LinkedHashMap
class LRUCache {
int cap;
LinkedHashMap<Integer,Integer> cache = new LinkedHashMap<>();
public LRUCache(int capacity) {
this.cap = capacity;
}
public int get(int key) {
if(!cache.containsKey(key)){
return -1;
}
makeRecently(key);
return cache.get(key);
}
public void put(int key, int value) {
if(cache.containsKey(key)){
cache.put(key,value);
makeRecently(key);
return;
}
if(cache.size()>=this.cap){
int oldestKey = cache.keySet().iterator().next();
cache.remove(oldestKey);
}
cache.put(key,value);
}
private void makeRecently (int key){
int value = cache.get(key);
cache.remove(key);
cache.put(key,value);
}
}
|