HyCodeYourTale
classpublicPriority 3

SizedTimeoutCache

com.hypixel.hytale.server.worldgen.util.cache.SizedTimeoutCache

implements Cache<K, V>

5

Methods

5

Public Methods

5

Fields

1

Constructors

Constructors

public
SizedTimeoutCache(long expire, TimeUnit unit, int maxSize, Function<K, V> func, BiConsumer<K, V> destroyer)

Methods

Public Methods (5)

public
void cleanup()
@Override
public
V get(K key)
@Nullable@Override
public
V getWithReusedKey(K reusedKey, Function<K, K> keyPool)
@Nullable
public
void put(K key, V value)
public
void shutdown()
@Override

Fields

Private/Package Fields (5)

privateCleanable cleanable
privateBiConsumer<K, V> destroyer
privateFunction<K, V> func
privateint maxSize
privatelong timeout

Inheritance

Parent
Current
Interface
Child

Use mouse wheel to zoom, drag to pan. Click nodes to navigate.

Related Classes

Source Code

package com.hypixel.hytale.server.worldgen.util.cache;

import com.hypixel.hytale.server.core.HytaleServer;
import it.unimi.dsi.fastutil.objects.Object2ObjectLinkedOpenHashMap;
import java.lang.ref.WeakReference;
import java.lang.ref.Cleaner.Cleanable;
import java.util.ArrayDeque;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import java.util.function.BiConsumer;
import java.util.function.Function;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;

public class SizedTimeoutCache<K, V> implements Cache<K, V> {
   private final ArrayDeque<SizedTimeoutCache.CacheEntry<K, V>> pool = new ArrayDeque<>();
   private final Object2ObjectLinkedOpenHashMap<K, SizedTimeoutCache.CacheEntry<K, V>> map = new Object2ObjectLinkedOpenHashMap();
   private final long timeout;
   private final int maxSize;
   @Nullable
   private final Function<K, V> func;
   @Nullable
   private final BiConsumer<K, V> destroyer;
   @Nonnull
   private final ScheduledFuture<?> future;
   @Nonnull
   private final Cleanable cleanable;

   public SizedTimeoutCache(long expire, @Nonnull TimeUnit unit, int maxSize, @Nullable Function<K, V> func, @Nullable BiConsumer<K, V> destroyer) {
      this.timeout = unit.toNanos(expire);
      this.maxSize = maxSize;
      this.func = func;
      this.destroyer = destroyer;
      this.future = HytaleServer.SCHEDULED_EXECUTOR.scheduleWithFixedDelay(new CleanupRunnable(new WeakReference<>(this)), expire, expire, unit);
      this.cleanable = CleanupFutureAction.CLEANER.register(this, new CleanupFutureAction(this.future));
   }

   @Override
   public void cleanup() {
      this.reduceLength(this.maxSize);
      long expire = System.nanoTime() - this.timeout;

      while (true) {
         K key;
         V value;
         synchronized (this.map) {
            label37: {
               if (!this.map.isEmpty()) {
                  key = (K)this.map.lastKey();
                  SizedTimeoutCache.CacheEntry<K, V> entry = (SizedTimeoutCache.CacheEntry<K, V>)this.map.get(key);
                  if (entry.timestamp <= expire) {
                     this.map.remove(key);
                     value = entry.value;
                     if (this.pool.size() < this.maxSize) {
                        entry.key = null;
                        entry.value = null;
                        entry.timestamp = 0L;
                        this.pool.addLast(entry);
                     }
                     break label37;
                  }
               }

               return;
            }
         }

         if (this.destroyer != null) {
            this.destroyer.accept(key, value);
         }
      }
   }

   private void reduceLength(int targetSize) {
      while (true) {
         K key;
         V value;
         synchronized (this.map) {
            if (this.map.size() <= targetSize) {
               return;
            }

            SizedTimeoutCache.CacheEntry<K, V> entry = (SizedTimeoutCache.CacheEntry<K, V>)this.map.removeLast();
            key = entry.key;
            value = entry.value;
            if (this.pool.size() < this.maxSize) {
               entry.key = null;
               entry.value = null;
               entry.timestamp = 0L;
               this.pool.addLast(entry);
            }
         }

         if (this.destroyer != null) {
            this.destroyer.accept(key, value);
         }
      }
   }

   @Override
   public void shutdown() {
      this.cleanable.clean();
      if (this.destroyer != null) {
         this.reduceLength(0);
      } else {
         synchronized (this.map) {
            this.map.clear();
         }
      }
   }

   @Nullable
   @Override
   public V get(K key) {
      if (this.future.isCancelled()) {
         throw new IllegalStateException("Cache has been shutdown!");
      } else {
         long timestamp = System.nanoTime();
         synchronized (this.map) {
            SizedTimeoutCache.CacheEntry<K, V> entry = (SizedTimeoutCache.CacheEntry<K, V>)this.map.getAndMoveToFirst(key);
            if (entry != null) {
               entry.timestamp = timestamp;
               return entry.value;
            }
         }

         if (this.func == null) {
            return null;
         } else {
            V value = this.func.apply(key);
            timestamp = System.nanoTime();
            SizedTimeoutCache.CacheEntry<K, V> resultEntry;
            V resultValue;
            SizedTimeoutCache.CacheEntry<K, V> newEntry;
            synchronized (this.map) {
               newEntry = this.pool.isEmpty() ? new SizedTimeoutCache.CacheEntry<>() : this.pool.removeLast();
               newEntry.key = key;
               newEntry.value = value;
               newEntry.timestamp = timestamp;
               resultEntry = (SizedTimeoutCache.CacheEntry<K, V>)this.map.getAndMoveToFirst(key);
               if (resultEntry != null) {
                  resultEntry.timestamp = timestamp;
               } else {
                  resultEntry = newEntry;
                  this.map.put(key, newEntry);
               }

               resultValue = resultEntry.value;
            }

            if (resultEntry != newEntry && this.destroyer != null) {
               this.destroyer.accept(key, value);
            }

            return resultValue;
         }
      }
   }

   public void put(K key, V value) {
      if (this.future.isCancelled()) {
         throw new IllegalStateException("Cache has been shutdown!");
      } else {
         long timestamp = System.nanoTime();
         SizedTimeoutCache.CacheEntry<K, V> oldEntry;
         synchronized (this.map) {
            SizedTimeoutCache.CacheEntry<K, V> entry = this.pool.isEmpty() ? new SizedTimeoutCache.CacheEntry<>() : this.pool.removeLast();
            entry.key = key;
            entry.value = value;
            entry.timestamp = timestamp;
            oldEntry = (SizedTimeoutCache.CacheEntry<K, V>)this.map.putAndMoveToFirst(key, entry);
            if (oldEntry != null) {
               entry.key = oldEntry.key;
            }
         }

         if (oldEntry != null && this.destroyer != null) {
            this.destroyer.accept(key, oldEntry.value);
         }
      }
   }

   @Nullable
   public V getWithReusedKey(K reusedKey, @Nonnull Function<K, K> keyPool) {
      if (this.future.isCancelled()) {
         throw new IllegalStateException("Cache has been shutdown!");
      } else {
         long timestamp = System.nanoTime();
         synchronized (this.map) {
            SizedTimeoutCache.CacheEntry<K, V> entry = (SizedTimeoutCache.CacheEntry<K, V>)this.map.getAndMoveToFirst(reusedKey);
            if (entry != null) {
               entry.timestamp = timestamp;
               return entry.value;
            }
         }

         if (this.func == null) {
            return null;
         } else {
            K newKey = keyPool.apply(reusedKey);
            V value = this.func.apply(newKey);
            timestamp = System.nanoTime();
            SizedTimeoutCache.CacheEntry<K, V> newEntry;
            SizedTimeoutCache.CacheEntry<K, V> resultEntry;
            V resultValue;
            synchronized (this.map) {
               newEntry = this.pool.isEmpty() ? new SizedTimeoutCache.CacheEntry<>() : this.pool.removeLast();
               newEntry.key = newKey;
               newEntry.value = value;
               newEntry.timestamp = timestamp;
               resultEntry = (SizedTimeoutCache.CacheEntry<K, V>)this.map.getAndMoveToFirst(newKey);
               if (resultEntry != null) {
                  resultEntry.timestamp = timestamp;
               } else {
                  resultEntry = newEntry;
                  this.map.put(newKey, newEntry);
               }

               resultValue = resultEntry.value;
            }

            if (resultEntry != newEntry && this.destroyer != null) {
               this.destroyer.accept(newKey, value);
            }

            return resultValue;
         }
      }
   }

   private static class CacheEntry<K, V> {
      @Nullable
      private V value;
      @Nullable
      private K key;
      private long timestamp;

      private CacheEntry() {
      }
   }
}