forked from gouthampradhan/leetcode
-
Notifications
You must be signed in to change notification settings - Fork 0
/
LRUCache.java
161 lines (146 loc) · 4.04 KB
/
LRUCache.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
package design;
import java.util.*;
/**
* Created by gouthamvidyapradhan on 18/03/2017.
Design and implement a data structure for Least Recently Used (LRU) cache. It should support the following operations: get and put.
get(key) - Get the value (will always be positive) of the key if the key exists in the cache, otherwise return -1.
put(key, value) - Set or insert the value if the key is not already present. When the cache reached its capacity, it should invalidate the least recently used item before inserting a new item.
Follow up:
Could you do both operations in O(1) time complexity?
Example:
LRUCache cache = new LRUCache( 2 /* capacity */ /*);
cache.put(1, 1);
cache.put(2, 2);
cache.get(1); // returns 1
cache.put(3, 3); // evicts key 2
cache.get(2); // returns -1 (not found)
cache.put(4, 4); // evicts key 1
cache.get(1); // returns -1 (not found)
cache.get(3); // returns 3
cache.get(4); // returns 4
Show Company Tags
Show Tags
Show Similar Problems
*/
public class LRUCache
{
public static class DLinkList
{
int key, value;
DLinkList left;
DLinkList right;
DLinkList(int key, int value)
{
this.key = key;
this.value = value;
left = null;
right = null;
}
}
private Map<Integer, DLinkList> cache;
private DLinkList head, tail;
private int capacity, currentSize;
/**
* Pop head node
* @return
*/
private DLinkList popHead()
{
if(!head.right.equals(tail))
{
DLinkList node = head.right;
head.right = node.right;
node.right.left = head;
node.right = null;
node.left = null;
return node;
}
return null;
}
/**
* Push to tail
* @param node
*/
private void offer(DLinkList node)
{
tail.left.right = node;
node.left = tail.left;
node.right = tail;
tail.left = node;
}
/**
* Move node to tail
* @param node
*/
private void moveToTail(DLinkList node)
{
node.left.right = node.right;
node.right.left = node.left;
offer(node);
}
/**
* Main method
* @param args
* @throws Exception
*/
public static void main(String[] args) throws Exception
{
LRUCache cache = new LRUCache(2);
cache.put(1, 1);
cache.put(2, 2);
System.out.println(cache.get(1));
cache.put(3,3);
System.out.println(cache.get(2));
cache.put(4, 4);
System.out.println(cache.get(1));
System.out.println(cache.get(3));
System.out.println(cache.get(4));
}
public LRUCache(int capacity)
{
this.capacity = capacity;
this.currentSize = 0;
cache = new HashMap<>();
head = new DLinkList(-1, -1);
tail = new DLinkList(-1, -1);
head.right = tail;
tail.left = head;
}
public int get(int key)
{
if(cache.get(key) == null) return -1;
DLinkList node = cache.get(key);
moveToTail(node);
return node.value;
}
public void put(int key, int value)
{
if(cache.containsKey(key))
{
DLinkList node = cache.get(key);
node.value = value;
moveToTail(node);
}
else
{
if(capacity == currentSize)
{
DLinkList head = popHead();
if(head != null)
{
cache.remove(head.key);
DLinkList node = new DLinkList(key, value);
offer(node);
cache.put(key, node);
}
}
else
{
DLinkList node = new DLinkList(key, value);
offer(node);
cache.put(key, node);
++currentSize;
}
}
}
}