package DataStructures.Caches;
import java.util.HashMap;
import java.util.Map;
/**
* Least recently used (LRU)
*
* Discards the least recently used items first.
* This algorithm requires keeping track of what was used when,
* which is expensive if one wants to make sure the algorithm always discards
* the least recently used item.
* https://en.wikipedia.org/wiki/Cache_replacement_policies#Least_recently_used_(LRU)
*
* @param key type
* @param value type
*/
public class LRUCache {
private final Map> data = new HashMap<>();
private Entry head;
private Entry tail;
private int cap;
private static final int DEFAULT_CAP = 100;
public LRUCache() {
setCapacity(DEFAULT_CAP);
}
public LRUCache(int cap) {
setCapacity(cap);
}
private void setCapacity(int newCapacity) {
checkCapacity(newCapacity);
for (int i = data.size(); i > newCapacity; i--) {
Entry evicted = evict();
data.remove(evicted.getKey());
}
this.cap = newCapacity;
}
private Entry evict() {
if (head == null) {
throw new RuntimeException("cache cannot be empty!");
}
Entry evicted = head;
head = evicted.getNextEntry();
head.setPreEntry(null);
evicted.setNextEntry(null);
return evicted;
}
private void checkCapacity(int capacity) {
if (capacity <= 0) {
throw new RuntimeException("capacity must greater than 0!");
}
}
public V get(K key) {
if (!data.containsKey(key)) {
return null;
}
final Entry entry = data.get(key);
moveNodeToLast(entry);
return entry.getValue();
}
private void moveNodeToLast(Entry entry) {
if (tail == entry) {
return;
}
final Entry preEntry = entry.getPreEntry();
final Entry nextEntry = entry.getNextEntry();
if (preEntry != null) {
preEntry.setNextEntry(nextEntry);
}
if (nextEntry != null) {
nextEntry.setPreEntry(preEntry);
}
if (head == entry) {
head = nextEntry;
}
tail.setNextEntry(entry);
entry.setPreEntry(tail);
entry.setNextEntry(null);
tail = entry;
}
public void put(K key, V value) {
if (data.containsKey(key)) {
final Entry existingEntry = data.get(key);
existingEntry.setValue(value);
moveNodeToLast(existingEntry);
return;
}
Entry newEntry;
if (data.size() == cap) {
newEntry = evict();
data.remove(newEntry.getKey());
} else {
newEntry = new Entry<>();
}
newEntry.setKey(key);
newEntry.setValue(value);
addNewEntry(newEntry);
data.put(key, newEntry);
}
private void addNewEntry(Entry newEntry) {
if (data.isEmpty()) {
head = newEntry;
tail = newEntry;
return;
}
tail.setNextEntry(newEntry);
newEntry.setPreEntry(tail);
newEntry.setNextEntry(null);
tail = newEntry;
}
static final class Entry {
private Entry preEntry;
private Entry nextEntry;
private I key;
private J value;
public Entry() {
}
public Entry(Entry preEntry, Entry nextEntry, I key, J value) {
this.preEntry = preEntry;
this.nextEntry = nextEntry;
this.key = key;
this.value = value;
}
public Entry getPreEntry() {
return preEntry;
}
public void setPreEntry(Entry preEntry) {
this.preEntry = preEntry;
}
public Entry getNextEntry() {
return nextEntry;
}
public void setNextEntry(Entry nextEntry) {
this.nextEntry = nextEntry;
}
public I getKey() {
return key;
}
public void setKey(I key) {
this.key = key;
}
public J getValue() {
return value;
}
public void setValue(J value) {
this.value = value;
}
}
public static void main(String[] args) {
final LRUCache cache = new LRUCache<>(2);
cache.put("Key1", 1);
cache.put("Key2", 2);
cache.put("Key3", 3);
cache.put("Key4", 4);
System.out.println("getValue(Key1): " + cache.get("Key1"));
System.out.println("getValue(Key2): " + cache.get("Key2"));
System.out.println("getValue(Key3): " + cache.get("Key3"));
System.out.println("getValue(Key4): " + cache.get("Key4"));
}
}