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

edu.stanford.nlp.util.CollectionValuedMap Maven / Gradle / Ivy

Go to download

Stanford Parser processes raw text in English, Chinese, German, Arabic, and French, and extracts constituency parse trees.

There is a newer version: 3.9.2
Show newest version
package edu.stanford.nlp.util;

import java.io.Serializable;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;

/**
 * Map from keys to {@link Collection}s. Important methods are the {@link #add}
 * and {@link #remove} methods for adding and removing a value to/from the
 * Collection associated with the key, and the {@link #get} method for getting
 * the Collection associated with a key. The class is quite general, because on
 * construction, it is possible to pass a {@link MapFactory} which will be used
 * to create the underlying map and a {@link CollectionFactory} which will be
 * used to create the Collections. Thus this class can be configured to act like
 * a "HashSetValuedMap" or a "ListValuedMap", or even a
 * "HashSetValuedIdentityHashMap". The possibilities are endless!
 *
 * @param  Key type of map
 * @param  Type of the Collection that is the Map's value
 * @author Teg Grenager ([email protected])
 * @author Sarah Spikes ([email protected]) - cleanup and filling in
 *         types
 */
public class CollectionValuedMap implements Map>, Serializable {

  private static final long serialVersionUID = -9064664153962599076L;

  @SuppressWarnings("serial")
  private final Map> map;
  protected final CollectionFactory cf;
  protected final boolean treatCollectionsAsImmutable;
  protected final MapFactory> mf;

  /**
   * Replaces current Collection mapped to key with the specified Collection.
   * Use carefully!
   */
  @Override
  public Collection put(K key, Collection collection) {
    return map.put(key, collection);
  }

  /**
   * Unsupported. Use {@link #addAll(Map)} instead.
   */
  @Override
  public void putAll(Map> m) {
    throw new UnsupportedOperationException();
  }

  /**
   * The empty collection to be returned when a {@code get} doesn't find
   * the key. The collection returned should be empty, such as
   * Collections.emptySet, for example.
   */
  @SuppressWarnings("serial")
  private final Collection emptyValue;

  /**
   * @return the Collection mapped to by key, never null, but may be empty.
   */
  @Override
  public Collection get(Object key) {
    Collection c = map.get(key);
    if (c == null) {
      c = emptyValue;
    }
    return c;
  }

  /**
   * Adds the value to the Collection mapped to by the key.
   */
  public void add(K key, V value) {
    if (treatCollectionsAsImmutable) {
      Collection newC = cf.newCollection();
      Collection c = map.get(key);
      if (c != null) {
        newC.addAll(c);
      }
      newC.add(value);
      map.put(key, newC); // replacing the old collection
    } else {
      Collection c = map.get(key);
      if (c == null) {
        c = cf.newCollection();
        map.put(key, c);
      }
      c.add(value); // modifying the old collection
    }
  }

  /**
   * Adds the values to the Collection mapped to by the key.
   */
  public void addAll(K key, Collection values) {
    if (treatCollectionsAsImmutable) {
      Collection newC = cf.newCollection();
      Collection c = map.get(key);
      if (c != null) {
        newC.addAll(c);
      }
      newC.addAll(values);
      map.put(key, newC); // replacing the old collection
    } else {
      Collection c = map.get(key);
      if (c == null) {
        c = cf.newCollection();
        map.put(key, c);
      }
      c.addAll(values); // modifying the old collection
    }
  }

  /** Just add the key (empty collection, but key is in the keySet). */
  public void addKey(K key) {
    Collection c = map.get(key);
    if (c == null) {
      c = cf.newCollection();
      map.put(key, c);
    }
  }

  /**
   * Adds all of the mappings in m to this CollectionValuedMap. If m is a
   * CollectionValuedMap, it will behave strangely. Use the constructor instead.
   */
  public void addAll(Map m) {
    if (m instanceof CollectionValuedMap) {
      throw new UnsupportedOperationException();
    }
    for (Map.Entry e : m.entrySet()) {
      add(e.getKey(), e.getValue());
    }
  }

  public void addAll(CollectionValuedMap cvm) {
    for (Entry> entry : cvm.entrySet()) {
      K key = entry.getKey();
      Collection currentCollection = get(key);
      Collection newValues = entry.getValue();
      if (treatCollectionsAsImmutable) {
        Collection newCollection = cf.newCollection();
        if (currentCollection != null) {
          newCollection.addAll(currentCollection);
        }
        newCollection.addAll(newValues);
        map.put(key, newCollection); // replacing the old collection
      } else {
        boolean needToAdd = false;
        if (currentCollection == emptyValue) {
          currentCollection = cf.newCollection();
          needToAdd = true;
        }
        currentCollection.addAll(newValues); // modifying the old collection
        if (needToAdd) {
          map.put(key, currentCollection);
        }
      }
    }
  }

  /**
   * Removes the mapping associated with this key from this Map.
   *
   * @return the Collection mapped to by this key.
   */
  @Override
  public Collection remove(Object key) {
    return map.remove(key);
  }

  /**
   * Removes the mappings associated with the keys from this map.
   *
   * @param keys They keys to remove
   */
  @SuppressWarnings("Convert2streamapi")
  public void removeAll(Collection keys) {
    for (K k : keys) {
      remove(k);
    }
  }

  /**
   * Removes the value from the Collection mapped to by this key, leaving the
   * rest of the collection intact.
   *
   * @param key The key to the Collection to remove the value from
   * @param value The value to remove
   */
  public void removeMapping(K key, V value) {
    if (treatCollectionsAsImmutable) {
      Collection c = map.get(key);
      if (c != null) {
        Collection newC = cf.newCollection();
        newC.addAll(c);
        newC.remove(value);
        map.put(key, newC);
      }

    } else {
      Collection c = get(key);
      c.remove(value);
    }
  }

  /**
   * Clears this Map.
   */
  @Override
  public void clear() {
    map.clear();
  }

  /**
   * @return true iff this key is in this map
   */
  @Override
  public boolean containsKey(Object key) {
    return map.containsKey(key);
  }

  /**
   * Unsupported.
   */
  @Override
  public boolean containsValue(Object value) {
    throw new UnsupportedOperationException();
  }

  /**
   * @return true iff this Map has no mappings in it.
   */
  @Override
  public boolean isEmpty() {
    return map.isEmpty();
  }

  /**
   * Each element of the Set is a Map.Entry object, where getKey() returns the
   * key of the mapping, and getValue() returns the Collection mapped to by the
   * key.
   *
   * @return a Set view of the mappings contained in this map.
   */
  @Override
  public Set>> entrySet() {
    return map.entrySet();
  }

  /**
   * @return a Set view of the keys in this Map.
   */
  @Override
  public Set keySet() {
    return map.keySet();
  }

  /**
   * The number of keys in this map.
   */
  @Override
  public int size() {
    return map.size();
  }

  /**
   * @return a collection of the values (really, a collection of values) in this
   *         Map
   */
  @Override
  public Collection> values() {
    return map.values();
  }

  @SuppressWarnings("Convert2streamapi")
  public Collection allValues() {
    Collection c = cf.newCollection();
    for (Collection c1 : map.values()) {
      c.addAll(c1);
    }
    return c;
  }

  /**
   * @return true iff o is a CollectionValuedMap, and each key maps to the a
   *         Collection of the same objects in o as it does in this
   *         CollectionValuedMap.
   */
  @Override
  public boolean equals(Object o) {
    if (this == o) {
      return true;
    }
    if (!(o instanceof CollectionValuedMap)) {
      return false;
    }

    CollectionValuedMap other = ErasureUtils.uncheckedCast(o);

    if (other.size() != size()) {
      return false;
    }

    try {
      for (Map.Entry> e : entrySet()) {
        K key = e.getKey();
        Collection value = e.getValue();
        if (value == null) {
          if (!(other.get(key) == null && other.containsKey(key))) {
            return false;
          }
        } else {
          if (!value.equals(other.get(key))) {
            return false;
          }
        }
      }
    } catch (ClassCastException | NullPointerException unused) {
      return false;
    }

    return true;
  }

  /**
   * @return the hashcode of the underlying Map
   */
  @Override
  public int hashCode() {
    return map.hashCode();
  }

  /**
   * Creates a "delta copy" of this Map, where only the differences
   * from the original Map are represented. (This typically assumes
   * that this map will no longer be changed.)
   */
  public CollectionValuedMap deltaCopy() {
    Map> deltaMap = new DeltaMap<>(this.map);
    return new CollectionValuedMap<>(null, cf, true, deltaMap);
  }

  /**
   * @return A String representation of this CollectionValuedMap, with special
   *         machinery to avoid recursion problems
   */
  @Override
  public String toString() {
    StringBuilder buf = new StringBuilder();
    buf.append('{');

    Iterator>> i = entrySet().iterator();
    while (i.hasNext()) {
      Map.Entry> e = i.next();
      K key = e.getKey();
      Collection value = e.getValue();
      buf.append(key == this ? "(this Map)" : key).append('=').append(value == this ? "(this Map)" : value);

      if (i.hasNext()) {
        buf.append(", ");
      }
    }

    buf.append('}');
    return buf.toString();
  }


  /**
   * Creates a new empty CollectionValuedMap.
   *
   * @param mf A MapFactory which will be used to generate the underlying Map
   * @param cf A CollectionFactory which will be used to generate the Collections
   *          in each mapping
   * @param treatCollectionsAsImmutable If true, forces this Map to create new a Collection every time a
   *          new value is added to or deleted from the Collection a mapping.
   */
  public CollectionValuedMap(MapFactory> mf, CollectionFactory cf,
                             boolean treatCollectionsAsImmutable) {
    this(mf, cf, treatCollectionsAsImmutable, null);
  }

  /**
   * Creates a new CollectionValuedMap.
   *
   * @param mf A MapFactory which will be used to generate the underlying Map
   * @param cf A CollectionFactory which will be used to generate the Collections
   *          in each mapping
   * @param treatCollectionsAsImmutable If true, forces this Map to create new a Collection every time a
   *          new value is added to or deleted from the Collection a mapping.
   * @param map An existing map to use rather than initializing one with mf. If this is non-null it is
   *            used to initialize the map rather than mf.
   */
  private CollectionValuedMap(MapFactory> mf, CollectionFactory cf,
                             boolean treatCollectionsAsImmutable,
                             Map> map) {
    if (cf == null) {
      throw new IllegalArgumentException();
    }
    if (mf == null && map == null) {
      throw new IllegalArgumentException();
    }
    this.mf = mf;
    this.cf = cf;
    this.treatCollectionsAsImmutable = treatCollectionsAsImmutable;
    this.emptyValue = cf.newEmptyCollection();
    if (map != null) {
      this.map = map;
    } else {
      this.map = Collections.synchronizedMap(mf.newMap());
    }
  }

  /**
   * Creates a new CollectionValuedMap with all of the mappings from cvm.
   *
   * @param cvm The CollectionValueMap to copy as this object.
   */
  public CollectionValuedMap(CollectionValuedMap cvm) {
    this.mf = cvm.mf;
    this.cf = cvm.cf;
    this.treatCollectionsAsImmutable = cvm.treatCollectionsAsImmutable;
    this.emptyValue = cvm.emptyValue;
    map = Collections.synchronizedMap(mf.newMap());
    for (Map.Entry> entry : cvm.map.entrySet()) {
      K key = entry.getKey();
      Collection c = entry.getValue();
      for (V value : c) {
        add(key, value);
      }
    }
  }

  /**
   * Creates a new empty CollectionValuedMap which uses a HashMap as the
   * underlying Map, and HashSets as the Collections in each mapping. Does not
   * treat Collections as immutable.
   */
  public CollectionValuedMap() {
    this(MapFactory.hashMapFactory(), CollectionFactory.hashSetFactory(), false);
  }

  /**
   * Creates a new empty CollectionValuedMap which uses a HashMap as the
   * underlying Map. Does not treat Collections as immutable.
   *
   * @param cf A CollectionFactory which will be used to generate the Collections
   *          in each mapping
   */
  public CollectionValuedMap(CollectionFactory cf) {
    this(MapFactory.hashMapFactory(), cf, false);
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy