All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.blogspot.mydailyjava.weaklockfree.WeakConcurrentMap Maven / Gradle / Ivy

package com.blogspot.mydailyjava.weaklockfree;

import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.atomic.AtomicLong;

/**
 * 

* A thread-safe map with weak keys. Entries are based on a key's system hash code and keys are considered * equal only by reference equality. *

* This class does not implement the {@link java.util.Map} interface because this implementation is incompatible * with the map contract. While iterating over a map's entries, any key that has not passed iteration is referenced non-weakly. */ public class WeakConcurrentMap extends AbstractWeakConcurrentMap> { /** * Lookup keys are cached thread-locally to avoid allocations on lookups. * This is beneficial as the JIT unfortunately can't reliably replace the {@link LookupKey} allocation * with stack allocations, even though the {@link LookupKey} does not escape. */ private static final ThreadLocal> LOOKUP_KEY_CACHE = new ThreadLocal>() { @Override protected LookupKey initialValue() { return new LookupKey(); } }; private static final AtomicLong ID = new AtomicLong(); private final Thread thread; private final boolean reuseKeys; /** * @param cleanerThread {@code true} if a thread should be started that removes stale entries. */ public WeakConcurrentMap(boolean cleanerThread) { this(cleanerThread, isPersistentClassLoader(LookupKey.class.getClassLoader())); } /** * Checks whether the provided {@link ClassLoader} may be unloaded like a web application class loader, for example. *

* If the class loader can't be unloaded, it is safe to use {@link ThreadLocal}s and to reuse the {@link LookupKey}. * Otherwise, the use of {@link ThreadLocal}s may lead to class loader leaks as it prevents the class loader this class * is loaded by to unload. *

* * @param classLoader The class loader to check. * @return {@code true} if the provided class loader can be unloaded. */ private static boolean isPersistentClassLoader(ClassLoader classLoader) { try { return classLoader == null // bootstrap class loader || classLoader == ClassLoader.getSystemClassLoader() || classLoader == ClassLoader.getSystemClassLoader().getParent(); // ext/platfrom class loader; } catch (Throwable ignored) { return false; } } /** * @param cleanerThread {@code true} if a thread should be started that removes stale entries. * @param reuseKeys {@code true} if the lookup keys should be reused via a {@link ThreadLocal}. * Note that setting this to {@code true} may result in class loader leaks. * See {@link #isPersistentClassLoader(ClassLoader)} for more details. */ public WeakConcurrentMap(boolean cleanerThread, boolean reuseKeys) { this(cleanerThread, reuseKeys, new ConcurrentHashMap, V>()); } /** * @param cleanerThread {@code true} if a thread should be started that removes stale entries. * @param reuseKeys {@code true} if the lookup keys should be reused via a {@link ThreadLocal}. * Note that setting this to {@code true} may result in class loader leaks. * See {@link #isPersistentClassLoader(ClassLoader)} for more details. * @param target ConcurrentMap implementation that this class wraps. */ public WeakConcurrentMap(boolean cleanerThread, boolean reuseKeys, ConcurrentMap, V> target) { super(target); this.reuseKeys = reuseKeys; if (cleanerThread) { thread = new Thread(this); thread.setName("weak-ref-cleaner-" + ID.getAndIncrement()); thread.setPriority(Thread.MIN_PRIORITY); thread.setDaemon(true); thread.start(); } else { thread = null; } } @Override @SuppressWarnings("unchecked") protected LookupKey getLookupKey(K key) { LookupKey lookupKey; if (reuseKeys) { lookupKey = (LookupKey) LOOKUP_KEY_CACHE.get(); } else { lookupKey = new LookupKey(); } return lookupKey.withValue(key); } @Override protected void resetLookupKey(LookupKey lookupKey) { lookupKey.reset(); } /** * @return The cleaner thread or {@code null} if no such thread was set. */ public Thread getCleanerThread() { return thread; } /* * A lookup key must only be used for looking up instances within a map. For this to work, it implements an identical contract for * hash code and equals as the WeakKey implementation. At the same time, the lookup key implementation does not extend WeakReference * and avoids the overhead that a weak reference implies. */ // can't use AutoClosable/try-with-resources as this project still supports Java 6 static final class LookupKey { private K key; private int hashCode; LookupKey withValue(K key) { this.key = key; hashCode = System.identityHashCode(key); return this; } /** * Failing to reset a lookup key can lead to memory leaks as the key is strongly referenced. */ void reset() { key = null; hashCode = 0; } @Override public boolean equals(Object other) { if (other instanceof WeakConcurrentMap.LookupKey) { return ((LookupKey) other).key == key; } else { return ((WeakKey) other).get() == key; } } @Override public int hashCode() { return hashCode; } } /** * A {@link WeakConcurrentMap} where stale entries are removed as a side effect of interacting with this map. */ public static class WithInlinedExpunction extends WeakConcurrentMap { public WithInlinedExpunction() { super(false); } @Override public V get(K key) { expungeStaleEntries(); return super.get(key); } @Override public boolean containsKey(K key) { expungeStaleEntries(); return super.containsKey(key); } @Override public V put(K key, V value) { expungeStaleEntries(); return super.put(key, value); } @Override public V remove(K key) { expungeStaleEntries(); return super.remove(key); } @Override public Iterator> iterator() { expungeStaleEntries(); return super.iterator(); } @Override public int approximateSize() { expungeStaleEntries(); return super.approximateSize(); } } }