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

ml.alternet.misc.SoftHashMap Maven / Gradle / Ivy

Go to download

Alternet Tools include discovery service tools, concurrent and locking tools, and more

The newest version!
package ml.alternet.misc;

import java.lang.ref.ReferenceQueue;
import java.lang.ref.SoftReference;
import java.util.AbstractMap;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.Map;
import java.util.Set;

/**
 * A soft map that discards its entry on memory leaks.
 *
 * @param 
 *            Key type
 * @param 
 *            Value type
 */
public class SoftHashMap extends AbstractMap {

    /** The internal HashMap that will hold the SoftReference. */
    private final Map hash = new HashMap<>();
    /** The number of "hard" references to hold internally. */
    private final int hardSize;
    /** The FIFO list of hard references, order of last access. */
    private final LinkedList hardCache = new LinkedList<>();
    /** Reference queue for cleared SoftReference objects. */
    private final ReferenceQueue queue = new ReferenceQueue<>();

    /**
     * Create a new soft hash map.
     */
    public SoftHashMap() {
        this(100);
    }

    /**
     * Numbers of items to keep in the map
     *
     * @param hardSize
     *            The number of items for which still exist a hard reference.
     */
    public SoftHashMap(int hardSize) {
        this.hardSize = hardSize;
    }

    @Override
    public V get(Object key) {
        V result = null;
        // We get the SoftReference represented by that key
        SoftEntry softRef = hash.get(key);
        if (softRef != null) {
            // From the SoftReference we get the value, which can be
            // null if it was not in the map, or it was removed in
            // the processQueue() method defined below
            result = softRef.get();
            if (result == null) {
                // If the value has been garbage collected, remove the
                // entry from the HashMap.
                hash.remove(key);
            } else {
                // We now add this object to the beginning of the hard
                // reference queue. One reference can occur more than
                // once, because lookups of the FIFO queue are slow, so
                // we don't want to search through it each time to remove
                // duplicates.
                hardCache.addFirst(result);
                if (hardCache.size() > hardSize) {
                    // Remove the last entry if list longer than HARD_SIZE
                    hardCache.removeLast();
                }
            }
        }
        return result;
    }

    /**
     * SoftReference which contains not only the value but also the key to make
     * it easier to find the entry in the HashMap after it's been garbage
     * collected.
     */
    private class SoftEntry extends SoftReference {
        private final Object key; // always make data member final

        private SoftEntry(V value, K key, ReferenceQueue q) {
            super(value, q);
            this.key = key;
        }
    }

    /**
     * Here we go through the ReferenceQueue and remove garbage collected
     * SoftValue objects from the HashMap by looking them up using the
     * SoftValue.key data member.
     */
    @SuppressWarnings("unchecked")
    private void processQueue() {
        SoftEntry sv;
        while ((sv = (SoftEntry) queue.poll()) != null) {
            hash.remove(sv.key); // we can access private data!
        }
    }

    /**
     * Here we put the key, value pair into the HashMap using a SoftValue
     * object.
     *
     * @param key The key
     * @param value The value
     * @return The previous value associated with key, or null
     *           if there was no mapping for key
     */
    @Override
    public V put(K key, V value) {
        processQueue(); // throw out garbage collected values first
        SoftEntry sv = hash.put(key, new SoftEntry(value, key, queue));
        if (sv == null) {
            return null;
        } else {
            return sv.get();
        }
    }

    @Override
    public V remove(Object key) {
        processQueue(); // throw out garbage collected values first
        SoftEntry sv = hash.remove(key);
        if (sv == null) {
            return null;
        } else {
            return sv.get();
        }
    }

    @Override
    public void clear() {
        hardCache.clear();
        processQueue(); // throw out garbage collected values
        hash.clear();
    }

    @Override
    public int size() {
        processQueue(); // throw out garbage collected values first
        return hash.size();
    }

    @Override
    public Set> entrySet() {
        throw new UnsupportedOperationException();
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy