WeakCache.java

  1. /*
  2.  *    Copyright 2009-2022 the original author or authors.
  3.  *
  4.  *    Licensed under the Apache License, Version 2.0 (the "License");
  5.  *    you may not use this file except in compliance with the License.
  6.  *    You may obtain a copy of the License at
  7.  *
  8.  *       http://www.apache.org/licenses/LICENSE-2.0
  9.  *
  10.  *    Unless required by applicable law or agreed to in writing, software
  11.  *    distributed under the License is distributed on an "AS IS" BASIS,
  12.  *    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13.  *    See the License for the specific language governing permissions and
  14.  *    limitations under the License.
  15.  */
  16. package org.apache.ibatis.cache.decorators;

  17. import java.lang.ref.ReferenceQueue;
  18. import java.lang.ref.WeakReference;
  19. import java.util.Deque;
  20. import java.util.LinkedList;

  21. import org.apache.ibatis.cache.Cache;

  22. /**
  23.  * Weak Reference cache decorator.
  24.  * Thanks to Dr. Heinz Kabutz for his guidance here.
  25.  *
  26.  * @author Clinton Begin
  27.  */
  28. public class WeakCache implements Cache {
  29.   private final Deque<Object> hardLinksToAvoidGarbageCollection;
  30.   private final ReferenceQueue<Object> queueOfGarbageCollectedEntries;
  31.   private final Cache delegate;
  32.   private int numberOfHardLinks;

  33.   public WeakCache(Cache delegate) {
  34.     this.delegate = delegate;
  35.     this.numberOfHardLinks = 256;
  36.     this.hardLinksToAvoidGarbageCollection = new LinkedList<>();
  37.     this.queueOfGarbageCollectedEntries = new ReferenceQueue<>();
  38.   }

  39.   @Override
  40.   public String getId() {
  41.     return delegate.getId();
  42.   }

  43.   @Override
  44.   public int getSize() {
  45.     removeGarbageCollectedItems();
  46.     return delegate.getSize();
  47.   }

  48.   public void setSize(int size) {
  49.     this.numberOfHardLinks = size;
  50.   }

  51.   @Override
  52.   public void putObject(Object key, Object value) {
  53.     removeGarbageCollectedItems();
  54.     delegate.putObject(key, new WeakEntry(key, value, queueOfGarbageCollectedEntries));
  55.   }

  56.   @Override
  57.   public Object getObject(Object key) {
  58.     Object result = null;
  59.     @SuppressWarnings("unchecked") // assumed delegate cache is totally managed by this cache
  60.     WeakReference<Object> weakReference = (WeakReference<Object>) delegate.getObject(key);
  61.     if (weakReference != null) {
  62.       result = weakReference.get();
  63.       if (result == null) {
  64.         delegate.removeObject(key);
  65.       } else {
  66.         synchronized (hardLinksToAvoidGarbageCollection) {
  67.           hardLinksToAvoidGarbageCollection.addFirst(result);
  68.           if (hardLinksToAvoidGarbageCollection.size() > numberOfHardLinks) {
  69.             hardLinksToAvoidGarbageCollection.removeLast();
  70.           }
  71.         }
  72.       }
  73.     }
  74.     return result;
  75.   }

  76.   @Override
  77.   public Object removeObject(Object key) {
  78.     removeGarbageCollectedItems();
  79.     @SuppressWarnings("unchecked")
  80.     WeakReference<Object> weakReference = (WeakReference<Object>) delegate.removeObject(key);
  81.     return weakReference == null ? null : weakReference.get();
  82.   }

  83.   @Override
  84.   public void clear() {
  85.     synchronized (hardLinksToAvoidGarbageCollection) {
  86.       hardLinksToAvoidGarbageCollection.clear();
  87.     }
  88.     removeGarbageCollectedItems();
  89.     delegate.clear();
  90.   }

  91.   private void removeGarbageCollectedItems() {
  92.     WeakEntry sv;
  93.     while ((sv = (WeakEntry) queueOfGarbageCollectedEntries.poll()) != null) {
  94.       delegate.removeObject(sv.key);
  95.     }
  96.   }

  97.   private static class WeakEntry extends WeakReference<Object> {
  98.     private final Object key;

  99.     private WeakEntry(Object key, Object value, ReferenceQueue<Object> garbageCollectionQueue) {
  100.       super(value, garbageCollectionQueue);
  101.       this.key = key;
  102.     }
  103.   }

  104. }