LRU implementation with expiration time

LRU implementation with expiration time

[[382833]]

I saw this algorithm a long time ago when I was studying operating systems. Later, I saw it more and more, and even saw it when I was reviewing the interview experience. To summarize:

1. What is LRU

LRU stands for Least Recently Used, which means the data that has not been used for the longest time. In other words, if a piece of data has not been used in the recent period, the chance of it being used in the future is also relatively small.

The common usage scenario is caching, such as the page replacement algorithm in the operating system. There are many implementation schemes. I have read many blogs, and most of them give four or five schemes. For the sake of simplicity, only one scheme with an expiration time is given here. Other implementations are similar, so I will leave it to you who are smart!!

Solution: Use linked list plus HashMap

Every time a new data comes, first check whether it is contained in the map. If so, move it to the head of the queue. If not, create a new node and put it in. For functions with expiration time, you only need to set an expiration time for each node, and delete it directly when this time is reached.

There is another problem: locks should be added in a multi-threaded environment. In order to ensure the flexibility of locks, we use ConcurrentHashMap.

OK, let's start implementing it:

2. Code Implementation

1. Define nodes

  1. //This Node uses each node in the HashMap
  2. class Node implements Comparable<Node> {
  3. private String key ;
  4. private Object value;
  5. private long expireTime; //Note that this expiration time is a time point, such as 11:11
  6. public Node(String key , Object value, long expireTime) {
  7. this.value = value;
  8. this.key = key ;
  9. this.expireTime = expireTime;
  10. }
  11. // Sort by expiration time
  12. @Override
  13. public   int compareTo(Node o) {
  14. long r = this.expireTime - o.expireTime;
  15. if (r > 0) return 1;
  16. if (r < 0) return -1;
  17. return 0;
  18. }
  19. }

2. LRU Implementation

  1. public class LRU {
  2. // Variable 1: used to set the thread pool for clearing expired data
  3. private static ScheduledExecutorService swapExpiredPool
  4. = new ScheduledThreadPoolExecutor(10);
  5. // Variable 2: User storage data. To ensure thread safety, ConcurrentHashMap is used.
  6. private ConcurrentHashMap<String, Node> cache = new ConcurrentHashMap<>(1024);
  7. // Variable 3: Save the latest expired data, the data with the shortest expiration time is placed at the front of the queue
  8. private PriorityQueue<Node> expireQueue = new PriorityQueue<>(1024);
  9. // Construction method: As long as there is a cache, the expired cleanup thread starts working
  10. public LRU() {
  11. swapExpiredPool.scheduleWithFixedDelay(new ExpiredNode(), 3,3,TimeUnit.SECONDS);
  12. }
  13. //And some code...
  14. }

Now we have defined several variables and a constructor, which means that once the LRU is started, it will start clearing. The clearing thread is ExpiredNode. Let's take a look:

3. Expired thread clearing method

This method is ExpiredNode, which is treated as an inner class in LRU.

  1. public class ExpiredNode implements Runnable {
  2. @Override
  3. public void run() {
  4. // Step 1: Get the current time
  5. long now = System.currentTimeMillis();
  6. while ( true ) {
  7. // Step 2: Pop the first element from the expired queue, and return if it does not exist or is not expired
  8. Node node = expireQueue.peek();
  9. if (node ​​== null || node.expireTime > now) return ;
  10. // Step 3: If it expires, delete it from the cache and pop it from the queue
  11. cache.remove( node.key );
  12. expireQueue.poll();
  13. }// This process is while( true ), and the judgment and deletion operations are performed continuously
  14. }
  15. }

Now that you know how to clear expired data, let's see how to add data.

4. set method

  1. public Object set (String key , Object value, long ttl) {
  2. // Step 1: Get the expiration time
  3. long expireTime = System.currentTimeMillis() + ttl;
  4. // Step 2: Create a new node
  5. Node newNode = new Node( key , value, expireTime);
  6. // Step 3: If there is one in the cache, overwrite it, if not, add a new one, and also add the expiration time queue
  7. Node old = cache.put( key , newNode);
  8. expireQueue.add (newNode);
  9. // Step 4: If the key has data, delete it from the expiration time queue
  10. if (old != null ) {
  11. expireQueue.remove(old);
  12. return old.value;
  13. }
  14. return   null ;
  15. }

5. get method

This method is relatively simple, you can get it directly.

  1. public Object get(String key ) {
  2. //Step 1: Get directly from cache. Note that this cache is a HashMap
  3. Node n = cache.get( key );
  4. //Step 2: If n is empty, it returns null , if it is not empty, it returns the corresponding value
  5. return n == null ? null : n.value;
  6. }

Note that the above 345 codes are all stored in LRU.

We already know the expiration time. In fact, we just add an expiration time queue and an expiration clearing thread. When clearing, use while(true) to determine whether the head of the queue is expired, and then determine whether to return and clear. When setting the method, we also need to add the new node to the queue and remove the old one. And we use ConcurrentHashMap to ensure thread safety.

OK, that’s all for today’s code.

This article is reprinted from the WeChat public account "Yugong Yaoyishan". You can follow it through the following QR code. To reprint this article, please contact the public account "Yugong Yaoyishan".

<<:  Can the United States' 6G layout surpass 5G and surpass my country?

>>:  Verizon installs 5G equipment outside homeowners' homes, sparking complaints

Blog    

Recommend

2018 Yunnan-Huawei Software Industry Summit was held on December 20

The 2018 Yunnan-Huawei Software Industry Summit w...

What exactly is “split brain” in distributed systems?

[[413929]] This article is reprinted from the WeC...

TCP three-way handshake and four-way wave

TCP (Transmission Control Protocol) is a connecti...

Interpretation of H3C's future industrial layout in the 5G era

One of the most important exhibitions of the year...

In the global 5G competition, who will be the ultimate beneficiary?

Leifeng.com: To understand cellular technology, y...

Detailed explanation of TCP/IP acceleration principle

Please look at this case first: For a certain key...

MEC – Are we getting closer?!

Multi-access edge computing (MEC) or previously m...