Skip to content

Commit 40d89e8

Browse files
committed
implement_lru_cache
1 parent e718fb4 commit 40d89e8

1 file changed

Lines changed: 84 additions & 0 deletions

File tree

Lines changed: 84 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
1+
class Node:
2+
__slots__ = ("key", "value", "prev", "next")
3+
4+
def __init__(self, key, value):
5+
self.key = key
6+
self.value = value
7+
self.prev = None
8+
self.next = None
9+
10+
11+
class LruCache:
12+
def __init__(self, limit: int):
13+
if limit < 1:
14+
raise ValueError(f"limit must be at least 1, got {limit}")
15+
16+
self._limit = limit
17+
self.map: dict = {}
18+
self.head = None
19+
self.tail = None
20+
21+
def _remove_node(self, node: Node):
22+
"""Remove a node from the linked list in O(1) time."""
23+
if node.prev:
24+
node.prev.next = node.next
25+
else:
26+
self.head = node.next
27+
28+
if node.next:
29+
node.next.prev = node.prev
30+
else:
31+
self.tail = node.prev
32+
33+
node.prev = None
34+
node.next = None
35+
36+
def _add_node_to_head(self, node: Node):
37+
"""Add a node to the head of the linked list in O(1) time."""
38+
node.next = self.head
39+
node.prev = None
40+
41+
if self.head:
42+
self.head.prev = node
43+
else:
44+
self.tail = node
45+
46+
self.head = node
47+
48+
def get(self, key):
49+
"""Return the value for key, Non if not present"""
50+
node = self.map.get(key)
51+
if node is None:
52+
return None
53+
54+
self._touch(node)
55+
return node.value
56+
57+
def set(self, key, value) -> None:
58+
"""Combine value with key, evicting the LRU entry if necessary."""
59+
node = self.map.get(key)
60+
61+
if node:
62+
node.value = value
63+
self._touch(node)
64+
else:
65+
if len(self.map) >= self._limit:
66+
self._evict()
67+
68+
new_node = Node(key, value)
69+
self._add_node_to_head(new_node)
70+
self.map[key] = new_node
71+
72+
def _touch(self, node) -> None:
73+
"""Move an existing node to the head (most-recently-used position)."""
74+
self._remove_node(node)
75+
self._add_node_to_head(node)
76+
77+
def _evict(self) -> None:
78+
"""Remove the least-recently-used entry (tail of the list)."""
79+
if self.tail is None:
80+
return
81+
82+
lru = self.tail
83+
self._remove_node(lru)
84+
del self.map[lru.key]

0 commit comments

Comments
 (0)