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

com.google.common.collect.StandardMultimap Maven / Gradle / Ivy

Go to download

Google Collections Library is a suite of new collections and collection-related goodness for Java 5.0

The newest version!
/*
 * Copyright (C) 2007 Google Inc.
 *
 * 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.google.common.collect;

import com.google.common.base.Nullable;
import com.google.common.base.Objects;

import static com.google.common.base.Preconditions.checkArgument;

import java.io.Serializable;
import java.util.AbstractCollection;
import java.util.AbstractMap;
import java.util.AbstractSet;
import java.util.Collection;
import java.util.Comparator;
import java.util.ConcurrentModificationException;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.RandomAccess;
import java.util.Set;
import java.util.SortedMap;
import java.util.SortedSet;

/**
 * Basic implementation of the {@link Multimap} interface. This class represents
 * a multimap as a map that associates each key with a collection of values. All
 * methods of {@link Multimap} are supported, including those specified as
 * optional in the interface.
 *
 * 

To implement a multimap, a subclass must define the method {@link * #createCollection()}, which creates an empty collection of values for a key. * *

The multimap constructor takes a map that has a single entry for each * distinct key. When you insert a key-value pair with a key that isn't already * in the multimap, {@code StandardMultimap} calls {@link #createCollection()} * to create the collection of values for that key. The subclass should not call * {@link #createCollection()} directly, and a new instance should be created * every time the method is called. * *

For example, the subclass could pass a {@link java.util.TreeMap} during * construction, and {@link #createCollection()} could return a {@link * java.util.TreeSet}, in which case the multimap's iterators would propagate * through the keys and values in sorted order. * *

Keys and values may be null, as long as the underlying collection classes * support null elements. * *

The collections created by {@link #createCollection()} may or may not * allow duplicates. If the collection, such as a {@link Set}, does not support * duplicates, an added key-value pair will replace an existing pair with the * same key and value, if such a pair is present. With collections like {@link * List} that allow duplicates, the collection will keep the existing key-value * pairs while adding a new pair. * *

This class is not threadsafe when any concurrent operations update the * multimap, even if the underlying map and {@link #createCollection()} method * return threadsafe classes. Concurrent read operations will work correctly. To * allow concurrent update operations, wrap your multimap with a call to {@link * Multimaps#synchronizedMultimap}. * * @author Jared Levy */ abstract class StandardMultimap implements Multimap, Serializable { /* * Here's an outline of the overall design. * * The map variable contains the collection of values associated with each * key. When a key-value pair is added to a multimap that didn't previously * contain and values for that key, a new collection generated by * createCollection is added to the map. That same collection instance * remains in the map as long as the multimap has any values for the key. If * all values for the key are removed, the key and collection are removed * from the map. * * The get method returns a WrappedCollection, which decorates the collection * in the map (if the key is present) or an empty collection (if the key is * not present). When the collection delegate in the WrappedCollection is * empty, the multimap may contain subsequently added values for that key. To * handle that situation, the WrappedCollection checks whether map contains * an entry for the provided key, and if so replaces the delegate. */ private final Map> map; private int totalSize; /** * Creates a new multimap that uses the provided map. * * @param map place to store the mapping from each key to its corresponding * values * @throws IllegalArgumentException if {@code map} is not empty */ protected StandardMultimap(Map> map) { checkArgument(map.isEmpty()); this.map = map; } /** * Creates the collection of values for a single key. * *

Collections with weak, soft, or phantom references are not supported. * Each call to {@code createCollection} should create a new instance. * *

The returned collection class determines whether duplicate key-value * pairs are allowed. * * @return an empty collection of values */ abstract Collection createCollection(); /** * Creates the collection of values for an explicitly provided key. By * default, it simply calls {@link #createCollection()}, which is the correct * behavior for most implementations. The {@link LinkedHashMultimap} class * overrides it. * * @param key key to associate with values in the collection * @return an empty collection of values */ Collection createCollection(@Nullable K key) { return createCollection(); } Map> backingMap() { return map; } // Query Operations public int size() { return totalSize; } public boolean isEmpty() { return totalSize == 0; } public boolean containsKey(@Nullable Object key) { return map.containsKey(key); } public boolean containsValue(@Nullable Object value) { for (Collection collection : map.values()) { if (collection.contains(value)) { return true; } } return false; } public boolean containsEntry(@Nullable Object key, @Nullable Object value) { Collection collection = map.get(key); return collection != null && collection.contains(value); } // Modification Operations public boolean put(@Nullable K key, @Nullable V value) { Collection collection = getOrCreateCollection(key); if (collection.add(value)) { totalSize++; return true; } else { return false; } } private Collection getOrCreateCollection(K key) { Collection collection = map.get(key); if (collection == null) { collection = createCollection(key); map.put(key, collection); } return collection; } public boolean remove(@Nullable Object key, @Nullable Object value) { Collection collection = map.get(key); if (collection == null) { return false; } boolean changed = collection.remove(value); if (changed) { totalSize--; if (collection.isEmpty()) { map.remove(key); } } return changed; } // Bulk Operations public void putAll(@Nullable K key, Iterable values) { if (!values.iterator().hasNext()) { return; } Collection collection = getOrCreateCollection(key); int oldSize = collection.size(); if (values instanceof Collection) { @SuppressWarnings("unchecked") Collection c = (Collection) values; collection.addAll(c); } else { for (V value : values) { collection.add(value); } } totalSize += (collection.size() - oldSize); } public void putAll(Multimap multimap) { for (Map.Entry entry : multimap.entries()) { put(entry.getKey(), entry.getValue()); } } public Collection replaceValues( @Nullable K key, Iterable values) { Iterator iterator = values.iterator(); if (!iterator.hasNext()) { return removeAll(key); } Collection collection = getOrCreateCollection(key); Collection oldValues = createCollection(); oldValues.addAll(collection); totalSize -= collection.size(); collection.clear(); while (iterator.hasNext()) { if (collection.add(iterator.next())) { totalSize++; } } return oldValues; } public Collection removeAll(@Nullable Object key) { Collection collection = map.remove(key); Collection output = createCollection(); if (collection != null) { output.addAll(collection); totalSize -= collection.size(); collection.clear(); } return output; } public void clear() { // Clear each collection, to make previously returned collections empty. for (Collection collection : map.values()) { collection.clear(); } map.clear(); totalSize = 0; } // Views /** * {@inheritDoc} * *

The returned collection is not serializable. */ public Collection get(@Nullable K key) { Collection collection = map.get(key); if (collection == null) { collection = createCollection(key); } return wrapCollection(key, collection); } /** * Generates a decorated collection that remains consistent with the values in * the multimap for the provided key. Changes to the multimap may alter the * returned collection, and vice versa. */ private Collection wrapCollection( @Nullable K key, Collection collection) { if (collection instanceof SortedSet) { return new WrappedSortedSet(key, (SortedSet) collection, null); } else if (collection instanceof Set) { return new WrappedSet(key, (Set) collection); } else if (collection instanceof List) { return wrapList(key, (List) collection, null); } else { return new WrappedCollection(key, collection, null); } } private List wrapList( @Nullable K key, List list, @Nullable WrappedCollection ancestor) { return (list instanceof RandomAccess) ? new RandomAccessWrappedList(key, list, ancestor) : new WrappedList(key, list, ancestor); } /** * Collection decorator that stays in sync with the multimap values for a key. * There are two kinds of wrapped collections: full and subcollections. Both * have a delegate pointing to the underlying collection class. * *

Full collections, identified by a null ancestor field, contain all * multimap values for a given key. Its delegate is a value in {@link * StandardMultimap#map} whenever the delegate is non-empty. The {@code * refreshIfEmpty}, {@code removeIfEmpty}, and {@code addToMap} methods ensure * that the WrappedCollection and map remain consistent. * *

A subcollection, such as a sublist, contains some of the values for a * given key. Its ancestor field points to the full wrapped collection with * all values for the key. The subcollection {@code refreshIfEmpty}, {@code * removeIfEmpty}, and {@code addToMap} methods call the corresponding methods * of the full wrapped collection. */ private class WrappedCollection extends AbstractCollection { final K key; Collection delegate; final WrappedCollection ancestor; final Collection originalAncestorDelegate; WrappedCollection(@Nullable K key, Collection delegate, @Nullable WrappedCollection ancestor) { this.key = key; this.delegate = delegate; this.ancestor = ancestor; this.originalAncestorDelegate = (ancestor == null) ? null : ancestor.getDelegate(); } /** * If the delegate collection is empty, but the multimap has values for the * key, replace the delegate with the new collection for the key. * *

For a subcollection, refresh its ancestor and validate that the * ancestor delegate hasn't changed. */ void refreshIfEmpty() { if (ancestor != null) { ancestor.refreshIfEmpty(); if (ancestor.getDelegate() != originalAncestorDelegate) { throw new ConcurrentModificationException(); } } else if (delegate.isEmpty()) { Collection newDelegate = map.get(key); if (newDelegate != null) { delegate = newDelegate; } } } /** * If collection is empty, remove it from {@code map}. For subcollections, * check whether the ancestor collection is empty. */ void removeIfEmpty() { if (ancestor != null) { ancestor.removeIfEmpty(); } else if (delegate.isEmpty()) { map.remove(key); } } K getKey() { return key; } /** * Add the delegate to the map. Other {@code WrappedCollection} methods * should call this method after adding elements to a previously empty * collection. * *

Subcollection add the ancestor's delegate instead. */ void addToMap() { if (ancestor != null) { ancestor.addToMap(); } else { map.put(key, delegate); } } @Override public int size() { refreshIfEmpty(); return delegate.size(); } @Override public boolean equals(Object other) { refreshIfEmpty(); return delegate.equals(other); } @Override public int hashCode() { refreshIfEmpty(); return delegate.hashCode(); } @Override public String toString() { refreshIfEmpty(); return delegate.toString(); } Collection getDelegate() { return delegate; } @Override public Iterator iterator() { refreshIfEmpty(); return new WrappedIterator(); } /** Collection iterator for {@code WrappedCollection}. */ class WrappedIterator implements Iterator { final Iterator delegateIterator; final Collection originalDelegate = delegate; WrappedIterator() { delegateIterator = iteratorOrListIterator(delegate); } WrappedIterator(Iterator delegateIterator) { this.delegateIterator = delegateIterator; } /** * If the delegate changed since the iterator was created, the iterator is * no longer valid. */ void validateIterator() { refreshIfEmpty(); if (delegate != originalDelegate) { throw new ConcurrentModificationException(); } } public boolean hasNext() { validateIterator(); return delegateIterator.hasNext(); } public V next() { validateIterator(); return delegateIterator.next(); } public void remove() { delegateIterator.remove(); totalSize--; removeIfEmpty(); } Iterator getDelegateIterator() { validateIterator(); return delegateIterator; } } @Override public boolean add(V value) { refreshIfEmpty(); boolean wasEmpty = delegate.isEmpty(); boolean changed = delegate.add(value); if (changed) { totalSize++; if (wasEmpty) { addToMap(); } } return changed; } WrappedCollection getAncestor() { return ancestor; } // The following methods are provided for better performance. @Override public boolean addAll(Collection collection) { if (collection.isEmpty()) { return false; } int oldSize = size(); // calls refreshIfEmpty boolean changed = delegate.addAll(collection); if (changed) { int newSize = delegate.size(); totalSize += (newSize - oldSize); if (oldSize == 0) { addToMap(); } } return changed; } @Override public boolean contains(Object o) { refreshIfEmpty(); return delegate.contains(o); } @Override public boolean containsAll(Collection c) { refreshIfEmpty(); return delegate.containsAll(c); } @Override public void clear() { int oldSize = size(); // calls refreshIfEmpty if (oldSize == 0) { return; } delegate.clear(); totalSize -= oldSize; removeIfEmpty(); // maybe shouldn't be removed if this is a sublist } @Override public boolean remove(Object o) { refreshIfEmpty(); boolean changed = delegate.remove(o); if (changed) { totalSize--; removeIfEmpty(); } return changed; } @Override public boolean removeAll(Collection c) { if (c.isEmpty()) { return false; } int oldSize = size(); // calls refreshIfEmpty boolean changed = delegate.removeAll(c); if (changed) { int newSize = delegate.size(); totalSize += (newSize - oldSize); removeIfEmpty(); } return changed; } @Override public boolean retainAll(Collection c) { int oldSize = size(); // calls refreshIfEmpty boolean changed = delegate.retainAll(c); if (changed) { int newSize = delegate.size(); totalSize += (newSize - oldSize); removeIfEmpty(); } return changed; } } private Iterator iteratorOrListIterator(Collection collection) { return (collection instanceof List) ? ((List) collection).listIterator() : collection.iterator(); } /** Set decorator that stays in sync with the multimap values for a key. */ private class WrappedSet extends WrappedCollection implements Set { WrappedSet(@Nullable K key, Set delegate) { super(key, delegate, null); } } /** * SortedSet decorator that stays in sync with the multimap values for a key. */ private class WrappedSortedSet extends WrappedCollection implements SortedSet { WrappedSortedSet(@Nullable K key, SortedSet delegate, @Nullable WrappedCollection ancestor) { super(key, delegate, ancestor); } SortedSet getSortedSetDelegate() { return (SortedSet) getDelegate(); } public Comparator comparator() { refreshIfEmpty(); return getSortedSetDelegate().comparator(); } public V first() { refreshIfEmpty(); return getSortedSetDelegate().first(); } public V last() { refreshIfEmpty(); return getSortedSetDelegate().last(); } public SortedSet headSet(V toElement) { refreshIfEmpty(); return new WrappedSortedSet( getKey(), getSortedSetDelegate().headSet(toElement), (getAncestor() == null) ? this : getAncestor()); } public SortedSet subSet(V fromElement, V toElement) { refreshIfEmpty(); return new WrappedSortedSet( getKey(), getSortedSetDelegate().subSet(fromElement, toElement), (getAncestor() == null) ? this : getAncestor()); } public SortedSet tailSet(V fromElement) { refreshIfEmpty(); return new WrappedSortedSet( getKey(), getSortedSetDelegate().tailSet(fromElement), (getAncestor() == null) ? this : getAncestor()); } } /** List decorator that stays in sync with the multimap values for a key. */ private class WrappedList extends WrappedCollection implements List { WrappedList(@Nullable K key, List delegate, @Nullable WrappedCollection ancestor) { super(key, delegate, ancestor); } List getListDelegate() { return (List) getDelegate(); } public boolean addAll(int index, Collection c) { if (c.isEmpty()) { return false; } int oldSize = size(); // calls refreshIfEmpty boolean changed = getListDelegate().addAll(index, c); if (changed) { int newSize = getDelegate().size(); totalSize += (newSize - oldSize); if (oldSize == 0) { addToMap(); } } return changed; } public V get(int index) { refreshIfEmpty(); return getListDelegate().get(index); } public V set(int index, V element) { refreshIfEmpty(); return getListDelegate().set(index, element); } public void add(int index, V element) { refreshIfEmpty(); boolean wasEmpty = getListDelegate().isEmpty(); getListDelegate().add(index, element); totalSize++; if (wasEmpty) { addToMap(); } } public V remove(int index) { refreshIfEmpty(); V value = getListDelegate().remove(index); totalSize--; removeIfEmpty(); return value; } public int indexOf(Object o) { refreshIfEmpty(); return getListDelegate().indexOf(o); } public int lastIndexOf(Object o) { refreshIfEmpty(); return getListDelegate().lastIndexOf(o); } public ListIterator listIterator() { refreshIfEmpty(); return new WrappedListIterator(); } public ListIterator listIterator(int index) { refreshIfEmpty(); return new WrappedListIterator(index); } public List subList(int fromIndex, int toIndex) { refreshIfEmpty(); return wrapList(getKey(), getListDelegate().subList(fromIndex, toIndex), (getAncestor() == null) ? this : getAncestor()); } /** ListIterator decorator. */ private class WrappedListIterator extends WrappedIterator implements ListIterator { WrappedListIterator() {} public WrappedListIterator(int index) { super(getListDelegate().listIterator(index)); } private ListIterator getDelegateListIterator() { return (ListIterator) getDelegateIterator(); } public boolean hasPrevious() { return getDelegateListIterator().hasPrevious(); } public V previous() { return getDelegateListIterator().previous(); } public int nextIndex() { return getDelegateListIterator().nextIndex(); } public int previousIndex() { return getDelegateListIterator().previousIndex(); } public void set(V value) { getDelegateListIterator().set(value); } public void add(V value) { boolean wasEmpty = isEmpty(); getDelegateListIterator().add(value); totalSize++; if (wasEmpty) { addToMap(); } } } } /** * List decorator that stays in sync with the multimap values for a key and * supports rapid random access. */ private class RandomAccessWrappedList extends WrappedList implements RandomAccess { RandomAccessWrappedList(@Nullable K key, List delegate, @Nullable WrappedCollection ancestor) { super(key, delegate, ancestor); } } private transient volatile KeySet keySet; public Set keySet() { if (keySet == null) { if (map instanceof SortedMap) { keySet = new SortedKeySet((SortedMap>) map); } else { keySet = new KeySet(map); } } return keySet; } private class KeySet extends AbstractSet { /** * This is usually the same as map, except when someone requests a * subcollection of a {@link SortedKeySet}. */ final Map> subMap; KeySet(final Map> subMap) { this.subMap = subMap; } @Override public int size() { return subMap.size(); } @Override public Iterator iterator() { return new Iterator() { final Iterator>> entryIterator = subMap.entrySet().iterator(); Map.Entry> entry; public boolean hasNext() { return entryIterator.hasNext(); } public K next() { entry = entryIterator.next(); return entry.getKey(); } public void remove() { entryIterator.remove(); totalSize -= entry.getValue().size(); entry.getValue().clear(); } }; } // The following methods are included for better performance. @Override public boolean contains(Object key) { return subMap.containsKey(key); } @Override public boolean remove(Object key) { int count = 0; Collection collection = subMap.remove(key); if (collection != null) { count = collection.size(); collection.clear(); totalSize -= count; } return count > 0; } @Override public boolean containsAll(Collection c) { return subMap.keySet().containsAll(c); } } private class SortedKeySet extends KeySet implements SortedSet { SortedKeySet(SortedMap> subMap) { super(subMap); } SortedMap> sortedMap() { return (SortedMap>) subMap; } public Comparator comparator() { return sortedMap().comparator(); } public K first() { return sortedMap().firstKey(); } public SortedSet headSet(K toElement) { return new SortedKeySet(sortedMap().headMap(toElement)); } public K last() { return sortedMap().lastKey(); } public SortedSet subSet(K fromElement, K toElement) { return new SortedKeySet(sortedMap().subMap(fromElement, toElement)); } public SortedSet tailSet(K fromElement) { return new SortedKeySet(sortedMap().tailMap(fromElement)); } } private transient volatile Multiset multiset; public Multiset keys() { if (multiset == null) { multiset = new MultisetView(); } return multiset; } /** Multiset view that stays in sync with the multimap keys. */ private class MultisetView extends AbstractMultiset { transient volatile Set> entrySet; @Override public int remove(Object key, int occurrences) { checkArgument(occurrences >= 0); Collection collection = map.get(key); if (collection == null) { return 0; } int count = collection.size(); if (occurrences >= count) { return removeValuesForKey(key); } Iterator iterator = collection.iterator(); for (int i = 0; i < occurrences; i++) { iterator.next(); iterator.remove(); } totalSize -= occurrences; return occurrences; } @Override public Set elementSet() { return StandardMultimap.this.keySet(); } @Override public Set> entrySet() { if (entrySet == null) { entrySet = new AbstractSet>() { @Override public Iterator> iterator() { return new MultisetEntryIterator(); } @Override public int size() { return map.size(); } // The following methods are included for better performance. @Override public boolean contains(Object o) { if (!(o instanceof Multiset.Entry)) { return false; } Multiset.Entry entry = (Multiset.Entry) o; Collection collection = map.get(entry.getElement()); return (collection != null) && (collection.size() == entry.getCount()); } @Override public void clear() { StandardMultimap.this.clear(); } @Override public boolean remove(Object o) { return contains(o) && (removeValuesForKey(((Multiset.Entry) o).getElement()) > 0); } }; } return entrySet; } @Override public Iterator iterator() { return new MultisetKeyIterator(); } // The following methods are included for better performance. @Override public int count(Object key) { Collection collection = map.get(key); return (collection == null) ? 0 : collection.size(); } @Override public int removeAllOccurrences(Object key) { return removeValuesForKey(key); } @Override public int size() { return totalSize; } } /** * Removes all values for the provided key. Unlike {@link #removeAll}, it * returns the number of removed mappings. */ private int removeValuesForKey(Object key) { int count = 0; Collection collection = map.remove(key); if (collection != null) { count = collection.size(); collection.clear(); totalSize -= count; } return count; } /** Iterator across each key, repeating once per value. */ private class MultisetEntryIterator implements Iterator> { final Iterator>> asMapIterator = asMap().entrySet().iterator(); public boolean hasNext() { return asMapIterator.hasNext(); } public Multiset.Entry next() { return new MultisetEntry(asMapIterator.next()); } public void remove() { asMapIterator.remove(); } } private class MultisetEntry extends AbstractMultisetEntry { final Map.Entry> entry; public MultisetEntry(Map.Entry> entry) { this.entry = entry; } public K getElement() { validate(); return entry.getKey(); } public int getCount() { validate(); return entry.getValue().size(); } private void validate() { if (entry.getValue().isEmpty()) { throw new ConcurrentModificationException(); } } } /** Iterator across each key, repeating once per value. */ private class MultisetKeyIterator implements Iterator { final Iterator> entryIterator = entries().iterator(); public boolean hasNext() { return entryIterator.hasNext(); } public K next() { return entryIterator.next().getKey(); } public void remove() { entryIterator.remove(); } } private transient volatile Collection values; /** * {@inheritDoc} * *

The iterator generated by the returned collection traverses the values * for one key, followed by the values of a second key, and so on. */ public Collection values() { if (values == null) { values = new AbstractCollection() { @Override public Iterator iterator() { return new ValueIterator(); } @Override public int size() { return totalSize; } // The following methods are included to improve performance. @Override public void clear() { StandardMultimap.this.clear(); } @Override public boolean contains(Object value) { return containsValue(value); } }; } return values; } /** Iterator across all values. */ private class ValueIterator implements Iterator { final Iterator> entryIterator = createEntryIterator(); public boolean hasNext() { return entryIterator.hasNext(); } public V next() { return entryIterator.next().getValue(); } public void remove() { entryIterator.remove(); } } private transient volatile Collection> entries; /** * {@inheritDoc} * *

The iterator generated by the returned collection traverses the values * for one key, followed by the values of a second key, and so on. */ public Collection> entries() { if (entries == null) { // TODO: can we refactor so we're not doing "this instanceof"? entries = (this instanceof SetMultimap) ? new EntrySet() : new Entries(); } return entries; } /** Entries for multimap. */ private class Entries extends AbstractCollection> { @Override public Iterator> iterator() { return createEntryIterator(); } @Override public int size() { return totalSize; } // The following methods are included to improve performance. @Override public boolean contains(Object o) { if (!(o instanceof Map.Entry)) { return false; } Map.Entry entry = (Map.Entry) o; return containsEntry(entry.getKey(), entry.getValue()); } @Override public void clear() { StandardMultimap.this.clear(); } @Override public boolean remove(Object o) { if (!(o instanceof Map.Entry)) { return false; } Map.Entry entry = (Map.Entry) o; return StandardMultimap.this.remove(entry.getKey(), entry.getValue()); } } /** * Returns an iterator across all key-value map entries, used by {@code * entries().iterator()} and {@code values().iterator()}. The default * behavior, which traverses the values for one key, the values for a second * key, and so on, suffices for most {@code StandardMultimap} implementations. * * @return an iterator across map entries */ Iterator> createEntryIterator() { return new EntryIterator(); } /** Iterator across all key-value pairs. */ private class EntryIterator implements Iterator> { final Iterator>> keyIterator; K key; Collection collection; Iterator valueIterator; EntryIterator() { keyIterator = map.entrySet().iterator(); if (keyIterator.hasNext()) { findValueIteratorAndKey(); } else { valueIterator = Iterators.emptyIterator(); } } void findValueIteratorAndKey() { Map.Entry> entry = keyIterator.next(); key = entry.getKey(); collection = entry.getValue(); valueIterator = iteratorOrListIterator(collection); } public boolean hasNext() { return valueIterator.hasNext() || keyIterator.hasNext(); } public Map.Entry next() { if (!hasNext()) { throw new NoSuchElementException(); } if (!valueIterator.hasNext()) { findValueIteratorAndKey(); } return (valueIterator instanceof ListIterator) ? new ListMapEntry(key, (ListIterator) valueIterator) : Maps.immutableEntry(key, valueIterator.next()); } public void remove() { valueIterator.remove(); if (collection.isEmpty()) { keyIterator.remove(); } totalSize--; } } /** * Map entry that allows value modification through provided list iterator. */ private class ListMapEntry extends AbstractMapEntry { final K key; V value; final ListIterator iterator; ListMapEntry(K key, ListIterator iterator) { this.key = key; this.value = iterator.next(); this.iterator = iterator; } @Override public K getKey() { return key; } @Override public V getValue() { return value; } @Override public V setValue(V newValue) { V oldValue = this.value; iterator.set(newValue); this.value = newValue; return oldValue; } } /** Entry set for a {@link SetMultimap}. */ private class EntrySet extends Entries implements Set> { @Override public boolean equals(Object other) { if (this == other) { return true; } if (!(other instanceof Set)) { return false; } Set otherSet = (Set) other; return (totalSize == otherSet.size()) && containsAll(otherSet); } @Override public int hashCode() { int hash = 0; for (Map.Entry entry : this) { hash += entry.hashCode(); } return hash; } } private transient volatile Map> asMap; public Map> asMap() { if (asMap == null) { asMap = new AbstractMap>() { volatile Set>> entrySet; @Override public Set>> entrySet() { if (entrySet == null) { entrySet = new AsMapEntries(); } return entrySet; } // The following methods are included for performance. @Override public void clear() { StandardMultimap.this.clear(); } @Override public boolean containsKey(Object key) { return map.containsKey(key); } @Override public Collection get(Object key) { Collection collection = map.get(key); if (collection == null) { return null; } @SuppressWarnings("unchecked") K k = (K) key; return wrapCollection(k, collection); } @Override public Collection remove(Object key) { Collection collection = removeAll(key); return collection.isEmpty() ? null : collection; } @Override public boolean equals(Object other) { return map.equals(other); } @Override public int hashCode() { return map.hashCode(); } @Override public String toString() { return map.toString(); } }; } return asMap; } private class AsMapEntries extends AbstractSet>> { @Override public Iterator>> iterator() { return new AsMapIterator(); } @Override public int size() { return map.size(); } // The following methods are included for performance. @Override public void clear() { StandardMultimap.this.clear(); } @Override public boolean contains(Object o) { if (!(o instanceof Map.Entry)) { return false; } Map.Entry entry = (Map.Entry) o; return (entry.getValue() != null) && Objects.equal(map.get(entry.getKey()), entry.getValue()); } @Override public boolean remove(Object o) { if (!(o instanceof Map.Entry)) { return false; } Map.Entry entry = (Map.Entry) o; return contains(entry) && (removeValuesForKey(entry.getKey()) > 0); } } /** Iterator across all keys and value collections. */ private class AsMapIterator implements Iterator>> { final Iterator>> delegateIterator = map.entrySet().iterator(); Collection collection; public boolean hasNext() { return delegateIterator.hasNext(); } public Map.Entry> next() { Map.Entry> entry = delegateIterator.next(); K key = entry.getKey(); collection = entry.getValue(); return Maps.immutableEntry(key, wrapCollection(key, collection)); } public void remove() { delegateIterator.remove(); totalSize -= collection.size(); collection.clear(); } } // Comparison and hashing @Override public boolean equals(@Nullable Object other) { if (this == other) { return true; } if (!(other instanceof Multimap)) { return false; } Multimap otherMultimap = (Multimap) other; return map.equals(otherMultimap.asMap()); } /** * Returns the hash code for this multimap. * *

The hash code of a multimap is defined as the hash code of the map view, * as returned by {@link Multimap#asMap}. * * @see Map#hashCode */ @Override public int hashCode() { return map.hashCode(); } /** * Returns a string representation of the multimap, generated by calling * {@code toString} on the map returned by {@link Multimap#asMap}. * * @return a string representation of the multimap */ @Override public String toString() { return map.toString(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy