from abc import abstractmethod import sys from collections import deque class LRUCache: """ Page Replacement Algorithm, Least Recently Used (LRU) Caching.""" dq_store = object() # Cache store of keys key_reference_map = object() # References of the keys in cache _MAX_CAPACITY: int = 10 # Maximum capacity of cache @abstractmethod def __init__(self, n: int): """ Creates an empty store and map for the keys. The LRUCache is set the size n. """ self.dq_store = deque() self.key_reference_map = set() if not n: LRUCache._MAX_CAPACITY = sys.maxsize elif n < 0: raise ValueError("n should be an integer greater than 0.") else: LRUCache._MAX_CAPACITY = n def refer(self, x): """ Looks for a page in the cache store and adds reference to the set. Remove the least recently used key if the store is full. Update store to reflect recent access. """ if x not in self.key_reference_map: if len(self.dq_store) == LRUCache._MAX_CAPACITY: last_element = self.dq_store.pop() self.key_reference_map.remove(last_element) else: index_remove = 0 for idx, key in enumerate(self.dq_store): if key == x: index_remove = idx break self.dq_store.remove(index_remove) self.dq_store.appendleft(x) self.key_reference_map.add(x) def display(self): """ Prints all the elements in the store. """ for k in self.dq_store: print(k) if __name__ == "__main__": lru_cache = LRUCache(4) lru_cache.refer(1) lru_cache.refer(2) lru_cache.refer(3) lru_cache.refer(1) lru_cache.refer(4) lru_cache.refer(5) lru_cache.display()