com.alibaba.toolkit.util.collection.SoftHashMap Maven / Gradle / Ivy
/*
* Copyright (c) 2002-2012 Alibaba Group Holding Limited.
* All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.alibaba.toolkit.util.collection;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.SoftReference;
import java.util.AbstractMap;
import java.util.AbstractSet;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set;
/**
* A memory-sensitive implementation of the Map
interface.
*
* 这个类是从sun的sun.misc.SoftCache
移植并修改的.
*
*
* A SoftCache
object uses {@link java.lang.ref.SoftReference soft
* references} to implement a memory-sensitive hash map. If the garbage
* collector determines at a certain point in time that a value object in a
* SoftCache
entry is no longer strongly reachable, then it may
* remove that entry in order to release the memory occupied by the value
* object. All SoftCache
objects are guaranteed to be completely
* cleared before the virtual machine will throw an
* OutOfMemoryError
. Because of this automatic clearing feature,
* the behavior of this class is somewhat different from that of other
* Map
implementations.
*
*
* Both null values and the null key are supported. This class has the same
* performance characteristics as the HashMap
class, and has the
* same efficiency parameters of initial capacity and
* load factor.
*
*
* Like most collection classes, this class is not synchronized. A synchronized
* SoftCache
may be constructed using the
* Collections.synchronizedMap
method.
*
*
* In typical usage this class will be subclassed and the fill
* method will be overridden. When the get
method is invoked on a
* key for which there is no mapping in the cache, it will in turn invoke the
* fill
method on that key in an attempt to construct a
* corresponding value. If the fill
method returns such a value
* then the cache will be updated and the new value will be returned. Thus, for
* example, a simple URL-content cache can be constructed as follows:
*
*
* public class URLCache extends SoftCache {
* protected Object fill(Object key) {
* return ((URL) key).getContent();
* }
* }
*
*
*
*
* The behavior of the SoftCache
class depends in part upon the
* actions of the garbage collector, so several familiar (though not required)
* Map
invariants do not hold for this class.
*
*
* Because entries are removed from a SoftCache
in response to
* dynamic advice from the garbage collector, a SoftCache
may
* behave as though an unknown thread is silently removing entries. In
* particular, even if you synchronize on a SoftCache
instance and
* invoke none of its mutator methods, it is possible for the size
* method to return smaller values over time, for the isEmpty
* method to return false
and then true
, for the
* containsKey
method to return true
and later
* false
for a given key, for the get
method to return
* a value for a given key but later return null
, for the
* put
method to return null
and the
* remove
method to return false
for a key that
* previously appeared to be in the map, and for successive examinations of the
* key set, the value set, and the entry set to yield successively smaller
* numbers of elements.
*
*
* @author Mark Reinhold
* @version 1.4, 00/02/02
* @see java.util.HashMap
* @see java.lang.ref.SoftReference
* @since JDK1.2
*/
public class SoftHashMap extends AbstractMap {
/*
* The basic idea of this implementation is to maintain an internal HashMap
* that maps keys to soft references whose referents are the keys' values;
* the various accessor methods dereference these soft references before
* returning values. Because we don't have access to the innards of the
* HashMap, each soft reference must contain the key that maps to it so that
* the processQueue method can remove keys whose values have been discarded.
* Thus the HashMap actually maps keys to instances of the ValueCell class,
* which is a simple extension of the SoftReference class.
*/
private static class ValueCell extends SoftReference {
private static Object INVALID_KEY = new Object();
private static int dropped = 0;
private Object key;
private ValueCell(Object key, Object value, ReferenceQueue queue) {
super(value, queue);
this.key = key;
}
private static ValueCell create(Object key, Object value, ReferenceQueue queue) {
if (value == null) {
return null;
}
return new ValueCell(key, value, queue);
}
private static Object strip(Object val, boolean drop) {
if (val == null) {
return null;
}
ValueCell vc = (ValueCell) val;
Object o = vc.get();
if (drop) {
vc.drop();
}
return o;
}
private boolean isValid() {
return key != INVALID_KEY;
}
private void drop() {
super.clear();
key = INVALID_KEY;
dropped++;
}
@Override
public boolean equals(Object obj) {
if (obj == null) {
return false;
}
if (obj == this) {
return true;
}
return valEquals(this.get(), ((ValueCell) obj).get());
}
@Override
public int hashCode() {
Object o = this.get();
return o == null ? 0 : o.hashCode();
}
}
/* Hash table mapping keys to ValueCells */
private Map hash;
/* Reference queue for cleared ValueCells */
private ReferenceQueue queue = new ReferenceQueue();
/*
* Process any ValueCells that have been cleared and enqueued by the garbage
* collector. This method should be invoked once by each public mutator in
* this class. We don't invoke this method in public accessors because that
* can lead to surprising ConcurrentModificationExceptions.
*/
private void processQueue() {
ValueCell vc;
while ((vc = (ValueCell) queue.poll()) != null) {
if (vc.isValid()) {
hash.remove(vc.key);
} else {
ValueCell.dropped--;
}
}
}
/* -- Constructors -- */
/**
* Construct a new, empty SoftCache
with the given initial
* capacity and the given load factor.
*
* @param initialCapacity The initial capacity of the cache
* @param loadFactor A number between 0.0 and 1.0
* @throws IllegalArgumentException If the initial capacity is less than or
* equal to zero, or if the load factor is less than zero
*/
public SoftHashMap(int initialCapacity, float loadFactor) {
hash = new HashMap(initialCapacity, loadFactor);
}
/**
* Construct a new, empty SoftCache
with the given initial
* capacity and the default load factor.
*
* @param initialCapacity The initial capacity of the cache
* @throws IllegalArgumentException If the initial capacity is less than or
* equal to zero
*/
public SoftHashMap(int initialCapacity) {
hash = new HashMap(initialCapacity);
}
/**
* Construct a new, empty SoftCache
with the default capacity
* and the default load factor.
*/
public SoftHashMap() {
hash = new HashMap();
}
/* -- Simple queries -- */
/**
* Return the number of key-value mappings in this cache. The time required
* by this operation is linear in the size of the map.
*/
@Override
public int size() {
return entrySet().size();
}
/** Return true
if this cache contains no key-value mappings. */
@Override
public boolean isEmpty() {
return entrySet().isEmpty();
}
/**
* Return true
if this cache contains a mapping for the
* specified key. If there is no mapping for the key, this method will not
* attempt to construct one by invoking the fill
method.
*
* @param key The key whose presence in the cache is to be tested
*/
@Override
public boolean containsKey(Object key) {
Object value = hash.get(key);
if (value == null) {
return hash.containsKey(key);
} else {
return ValueCell.strip(value, false) != null;
}
}
/* -- Lookup and modification operations -- */
/**
* Create a value object for the given key
. This method is
* invoked by the get
method when there is no entry for
* key
. If this method returns a non-null
value,
* then the cache will be updated to map key
to that value, and
* that value will be returned by the get
method.
*
* The default implementation of this method simply returns
* null
for every key
value. A subclass may
* override this method to provide more useful behavior.
*
*
* @param key The key for which a value is to be computed
* @return A value for key
, or null
if one could
* not be computed
* @see #get
*/
protected Object fill(Object key) {
return null;
}
/**
* Return the value to which this cache maps the specified key
.
* If the cache does not presently contain a value for this key, then invoke
* the fill
method in an attempt to compute such a value. If
* that method returns a non-null
value, then update the cache
* and return the new value. Otherwise, return null
.
*
* Note that because this method may update the cache, it is considered a
* mutator and may cause ConcurrentModificationException
s to be
* thrown if invoked while an iterator is in use.
*
*
* @param key The key whose associated value, if any, is to be returned
* @see #fill
*/
@Override
public Object get(Object key) {
processQueue();
Object v = hash.get(key);
if (v == null) {
v = fill(key);
if (v != null) {
hash.put(key, ValueCell.create(key, v, queue));
return v;
}
}
return ValueCell.strip(v, false);
}
/**
* Update this cache so that the given key
maps to the given
* value
. If the cache previously contained a mapping for
* key
then that mapping is replaced and the old value is
* returned.
*
* @param key The key that is to be mapped to the given value
* @param value The value to which the given key
is to be
* mapped
* @return The previous value to which this key was mapped, or
* null
if if there was no mapping for the key
*/
@Override
public Object put(Object key, Object value) {
processQueue();
ValueCell vc = ValueCell.create(key, value, queue);
return ValueCell.strip(hash.put(key, vc), true);
}
/**
* Remove the mapping for the given key
from this cache, if
* present.
*
* @param key The key whose mapping is to be removed
* @return The value to which this key was mapped, or null
if
* there was no mapping for the key
*/
@Override
public Object remove(Object key) {
processQueue();
return ValueCell.strip(hash.remove(key), true);
}
/** Remove all mappings from this cache. */
@Override
public void clear() {
processQueue();
hash.clear();
}
/* -- Views -- */
private static boolean valEquals(Object o1, Object o2) {
return o1 == null ? o2 == null : o1.equals(o2);
}
/*
* Internal class for entries. Because it uses SoftCache.this.queue, this
* class cannot be static.
*/
private class Entry implements Map.Entry {
private Map.Entry ent;
private Object value;/*
* Strong reference to value, to prevent the GC
* from flushing the value while this Entry exists
*/
Entry(Map.Entry ent, Object value) {
this.ent = ent;
this.value = value;
}
public Object getKey() {
return ent.getKey();
}
public Object getValue() {
return value;
}
public Object setValue(Object value) {
return ent.setValue(ValueCell.create(ent.getKey(), value, queue));
}
@Override
public boolean equals(Object o) {
if (!(o instanceof Map.Entry)) {
return false;
}
Map.Entry e = (Map.Entry) o;
return valEquals(ent.getKey(), e.getKey()) && valEquals(value, e.getValue());
}
@Override
public int hashCode() {
Object k;
return ((k = getKey()) == null ? 0 : k.hashCode()) ^ (value == null ? 0 : value.hashCode());
}
}
/* Internal class for entry sets */
private class EntrySet extends AbstractSet {
Set hashEntries = hash.entrySet();
@Override
public Iterator iterator() {
return new Iterator() {
Iterator hashIterator = hashEntries.iterator();
Entry next = null;
public boolean hasNext() {
while (hashIterator.hasNext()) {
Map.Entry ent = (Map.Entry) hashIterator.next();
ValueCell vc = (ValueCell) ent.getValue();
Object v = null;
if (vc != null && (v = vc.get()) == null) {
/* Value has been flushed by GC */
continue;
}
next = new Entry(ent, v);
return true;
}
return false;
}
public Object next() {
if (next == null && !hasNext()) {
throw new NoSuchElementException();
}
Entry e = next;
next = null;
return e;
}
public void remove() {
hashIterator.remove();
}
};
}
@Override
public boolean isEmpty() {
return !iterator().hasNext();
}
@Override
public int size() {
int j = 0;
for (Iterator i = iterator(); i.hasNext(); i.next()) {
j++;
}
return j;
}
@Override
public boolean remove(Object o) {
processQueue();
if (o instanceof Entry) {
return hashEntries.remove(((Entry) o).ent);
} else if (o instanceof Map.Entry) {
Map.Entry e = (Map.Entry) o;
return hashEntries.remove(new DefaultMapEntry(e.getKey(), ValueCell.create(e.getKey(), e.getValue(),
queue)));
} else {
return false;
}
}
}
private Set entrySet = null;
/** Return a Set
view of the mappings in this cache. */
@Override
public Set entrySet() {
if (entrySet == null) {
entrySet = new EntrySet();
}
return entrySet;
}
}