From 36b0d96e246ce4842689e2fe36f3a201dd6b4569 Mon Sep 17 00:00:00 2001 From: Nataliia Volkova Date: Sun, 15 Feb 2026 10:29:21 +0000 Subject: [PATCH 1/2] LruCache --- Sprint-2/implement_lru_cache/lru_cache.py | 73 +++++++++++++++++++++++ 1 file changed, 73 insertions(+) diff --git a/Sprint-2/implement_lru_cache/lru_cache.py b/Sprint-2/implement_lru_cache/lru_cache.py index e69de29..637355d 100644 --- a/Sprint-2/implement_lru_cache/lru_cache.py +++ b/Sprint-2/implement_lru_cache/lru_cache.py @@ -0,0 +1,73 @@ +class Node: + def __init__(self, key, value): + self.key = key + self.value = value + self.next = None + self.previous = None + +class LruCache: + def __init__(self, limit: int): + if limit <= 0: + raise ValueError("limit must be > 0") + + self.head = None + self.tail = None + self.limit = limit + self.dict = {} + + def set(self, key, value): + node = Node(key, value) + + self.dict[key] = node + self.add_to_head(node) + + if len(self.dict) > self.limit: + self.evict_tail() + + + def get(self, key): + node = self.dict.get(key) + + if node is None: + return None + else: + self.move_to_head(node) + return node.value + + def add_to_head(self, node): + node.previous = None + node.next = self.head + + if self.head is None: + self.head = self.tail = node + else: + self.head.previous = node + self.head = node + + def remove(self, node): + if node.previous is None: + self.head = node.next + else: + node.previous.next = node.next + + if node.next is None: + self.tail = node.previous + else: + node.next.previous = node.previous + + node.previous = None + node.next = None + + def move_to_head(self, node): + if node is self.head: + return + self.remove(node) + self.add_to_head(node) + + def evict_tail(self): + if self.tail is None: + return + + old_tail = self.tail + self.remove(old_tail) + del self.dict[old_tail.key] \ No newline at end of file From aa1cc838a6e1e45d5ac51e41864af006cd23fad5 Mon Sep 17 00:00:00 2001 From: Nataliia Volkova Date: Mon, 20 Apr 2026 21:33:33 +0100 Subject: [PATCH 2/2] implementation Single-Responsibility Principle --- Sprint-2/implement_lru_cache/lru_cache.py | 72 +++++++++++++---------- 1 file changed, 41 insertions(+), 31 deletions(-) diff --git a/Sprint-2/implement_lru_cache/lru_cache.py b/Sprint-2/implement_lru_cache/lru_cache.py index 637355d..5b629b1 100644 --- a/Sprint-2/implement_lru_cache/lru_cache.py +++ b/Sprint-2/implement_lru_cache/lru_cache.py @@ -5,34 +5,10 @@ def __init__(self, key, value): self.next = None self.previous = None -class LruCache: - def __init__(self, limit: int): - if limit <= 0: - raise ValueError("limit must be > 0") - +class List: + def __init__(self): self.head = None self.tail = None - self.limit = limit - self.dict = {} - - def set(self, key, value): - node = Node(key, value) - - self.dict[key] = node - self.add_to_head(node) - - if len(self.dict) > self.limit: - self.evict_tail() - - - def get(self, key): - node = self.dict.get(key) - - if node is None: - return None - else: - self.move_to_head(node) - return node.value def add_to_head(self, node): node.previous = None @@ -64,10 +40,44 @@ def move_to_head(self, node): self.remove(node) self.add_to_head(node) - def evict_tail(self): - if self.tail is None: + def pop_tail(self): + if not self.tail: + return None + old_tail=self.tail + self.remove(old_tail) + return old_tail + +class LruCache: + def __init__(self, limit: int): + if limit <= 0: + raise ValueError("limit must be > 0") + + self.limit = limit + self.cache = {} + self.list = List() + + def set(self, key, value): + node = self.cache.get(key) + + if node: + node.value=value + self.list.move_to_head(node) return + + node = Node(key, value) + self.cache[key] = node + self.list.add_to_head(node) - old_tail = self.tail - self.remove(old_tail) - del self.dict[old_tail.key] \ No newline at end of file + if len(self.cache) >= self.limit: + tail=self.list.pop_tail() + if tail: + del self.cache[tail.key] + + def get(self, key): + node = self.cache.get(key) + + if node is None: + return None + else: + self.list.move_to_head(node) + return node.value