com.helger.commons.collection.map.AbstractSoftMap Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of ph-commons Show documentation
Show all versions of ph-commons Show documentation
Java 1.8+ Library with tons of utility classes required in all projects
/*
* Copyright (C) 2014-2024 Philip Helger (www.helger.com)
* philip[at]helger[dot]com
*
* 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.helger.commons.collection.map;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.SoftReference;
import java.util.AbstractMap;
import java.util.Collection;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import com.helger.commons.ValueEnforcer;
import com.helger.commons.annotation.CodingStyleguideUnaware;
import com.helger.commons.annotation.OverrideOnDemand;
import com.helger.commons.annotation.ReturnsMutableObject;
import com.helger.commons.collection.impl.ICommonsMap;
import com.helger.commons.lang.GenericReflection;
/**
* Soft {@link Map} implementation based on
* http://www.javaspecialists.eu/archive/Issue015.html
* The entrySet
implementation is from
* org.hypergraphdb.util
* Note: {@link AbstractSoftMap} is NOT serializable!
*
* @author Philip Helger
* @param
* Key type
* @param
* Value type
*/
public abstract class AbstractSoftMap extends AbstractMap implements ICommonsMap
{
/**
* We define our own subclass of 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.
*
* @param
* Key type
* @param
* Value type
*/
protected static final class SoftValue extends SoftReference
{
private final K m_aKey;
private SoftValue (final K aKey, final V aValue, @Nullable final ReferenceQueue super V> aQueue)
{
super (aValue, aQueue);
m_aKey = aKey;
}
}
private static class SoftMapEntry implements Map.Entry >
{
private final K m_aKey;
private SoftValue m_aSoftValue;
public SoftMapEntry (final K aKey, final V aValue, @Nullable final ReferenceQueue aQueue)
{
m_aKey = aKey;
m_aSoftValue = new SoftValue <> (aKey, aValue, aQueue);
}
public K getKey ()
{
return m_aKey;
}
@Nonnull
public final SoftValue getValue ()
{
return m_aSoftValue;
}
@Nonnull
public final SoftValue setValue (@Nonnull final SoftValue aNew)
{
final SoftValue aOld = aNew;
m_aSoftValue = aNew;
return aOld;
}
}
private static final class SoftEntrySet implements Set >
{
private static final class SoftIterator implements Iterator >
{
private final Iterator >> m_aSrcIter;
private SoftIterator (@Nonnull final Iterator >> aSrcIter)
{
m_aSrcIter = aSrcIter;
}
public boolean hasNext ()
{
return m_aSrcIter.hasNext ();
}
public Map.Entry next ()
{
final Map.Entry > e = m_aSrcIter.next ();
return new MapEntry <> (e.getKey (), e.getValue ().get ());
}
@Override
public void remove ()
{
m_aSrcIter.remove ();
}
}
@CodingStyleguideUnaware
private final Set >> m_aSrcEntrySet;
private final ReferenceQueue m_aQueue;
private SoftEntrySet (@Nonnull final Set >> aSrcEntrySet,
@Nonnull final ReferenceQueue aQueue)
{
m_aSrcEntrySet = aSrcEntrySet;
m_aQueue = aQueue;
}
public boolean add (@Nonnull final Map.Entry aEntry)
{
return m_aSrcEntrySet.add (new SoftMapEntry <> (aEntry.getKey (), aEntry.getValue (), m_aQueue));
}
public boolean addAll (@Nonnull final Collection extends Map.Entry > c)
{
boolean result = false;
for (final Map.Entry e : c)
if (this.add (e))
result = true;
return result;
}
public void clear ()
{
m_aSrcEntrySet.clear ();
}
public boolean contains (final Object aEntryObj)
{
if (!(aEntryObj instanceof Map.Entry))
return false;
final Map.Entry aEntry = GenericReflection.uncheckedCast (aEntryObj);
return m_aSrcEntrySet.contains (new SoftMapEntry <> (aEntry.getKey (), aEntry.getValue (), m_aQueue));
}
public boolean containsAll (@Nonnull final Collection > c)
{
for (final Object x : c)
if (!this.contains (x))
return false;
return true;
}
public boolean isEmpty ()
{
return m_aSrcEntrySet.isEmpty ();
}
@Nonnull
public Iterator > iterator ()
{
final Iterator >> aSrcIter = m_aSrcEntrySet.iterator ();
return new SoftIterator <> (aSrcIter);
}
public boolean remove (final Object aEntryObj)
{
if (!(aEntryObj instanceof Map.Entry , ?>))
return false;
final Map.Entry aEntry = GenericReflection.uncheckedCast (aEntryObj);
return m_aSrcEntrySet.remove (new SoftMapEntry <> (aEntry.getKey (), aEntry.getValue (), m_aQueue));
}
public boolean removeAll (final Collection > c)
{
boolean result = false;
for (final Object x : c)
if (this.remove (x))
result = true;
return result;
}
public boolean retainAll (final Collection > c)
{
throw new UnsupportedOperationException ();
}
public int size ()
{
return m_aSrcEntrySet.size ();
}
@Nonnull
public Object [] toArray ()
{
return toArray (new MapEntry , ?> [0]);
}
@Nonnull
public T [] toArray (@Nullable final T [] a)
{
MapEntry [] result = null;
if (a instanceof MapEntry , ?> [] && a.length >= size ())
result = GenericReflection.uncheckedCast (a);
else
result = GenericReflection.uncheckedCast (new MapEntry , ?> [size ()]);
final Object [] aSrcArray = m_aSrcEntrySet.toArray ();
for (int i = 0; i < aSrcArray.length; i++)
{
final Map.Entry > e = GenericReflection.uncheckedCast (aSrcArray[i]);
result[i] = new MapEntry <> (e.getKey (), e.getValue ().get ());
}
return GenericReflection.uncheckedCast (result);
}
}
/** The internal HashMap that will hold the SoftReference. */
@CodingStyleguideUnaware
protected final Map > m_aSrcMap;
/** Reference queue for cleared SoftReference objects. */
private final ReferenceQueue m_aQueue = new ReferenceQueue <> ();
protected AbstractSoftMap (@Nonnull final Map > aSrcMap)
{
m_aSrcMap = ValueEnforcer.notNull (aSrcMap, "SrcMap");
}
/**
* Callback method invoked after a map entry is garbage collected
*
* @param aKey
* Key the removed key
*/
@OverrideOnDemand
protected void onEntryRemoved (final K aKey)
{}
@Override
public V get (final Object aKey)
{
V ret = null;
// We get the SoftReference represented by that key
final SoftValue aSoftValue = m_aSrcMap.get (aKey);
if (aSoftValue != 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
ret = aSoftValue.get ();
if (ret == null)
{
// If the value has been garbage collected, remove the
// entry from the HashMap.
if (m_aSrcMap.remove (aKey) != null)
onEntryRemoved (GenericReflection.uncheckedCast (aKey));
}
}
return ret;
}
/**
* Here we go through the ReferenceQueue and remove garbage collected
* SoftValue objects from the HashMap by looking them up using the
* SoftValue.m_aKey data member.
*/
private void _processQueue ()
{
SoftValue aSoftValue;
while ((aSoftValue = GenericReflection.uncheckedCast (m_aQueue.poll ())) != null)
{
m_aSrcMap.remove (aSoftValue.m_aKey);
}
}
/**
* Here we put the key, value pair into the HashMap using a SoftValue object.
*/
@Override
public V put (final K aKey, final V aValue)
{
// throw out garbage collected values first
_processQueue ();
final SoftValue aOld = m_aSrcMap.put (aKey, new SoftValue <> (aKey, aValue, m_aQueue));
return aOld == null ? null : aOld.get ();
}
@Override
public V remove (final Object aKey)
{
// throw out garbage collected values first
_processQueue ();
final SoftValue aRemoved = m_aSrcMap.remove (aKey);
return aRemoved == null ? null : aRemoved.get ();
}
@Override
public void clear ()
{
// throw out garbage collected values
_processQueue ();
m_aSrcMap.clear ();
}
@Override
public int size ()
{
// throw out garbage collected values first
_processQueue ();
return m_aSrcMap.size ();
}
@Override
@ReturnsMutableObject ("design")
@CodingStyleguideUnaware
public Set > entrySet ()
{
final Set >> aSrcEntrySet = m_aSrcMap.entrySet ();
return new SoftEntrySet <> (aSrcEntrySet, m_aQueue);
}
@Override
public boolean equals (final Object o)
{
if (o == this)
return true;
if (o == null || !getClass ().equals (o.getClass ()))
return false;
return super.equals (o);
}
@Override
public int hashCode ()
{
return super.hashCode ();
}
}