|  | /* | 
|  | * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. | 
|  | * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. | 
|  | * | 
|  | * This code is free software; you can redistribute it and/or modify it | 
|  | * under the terms of the GNU General Public License version 2 only, as | 
|  | * published by the Free Software Foundation.  Oracle designates this | 
|  | * particular file as subject to the "Classpath" exception as provided | 
|  | * by Oracle in the LICENSE file that accompanied this code. | 
|  | * | 
|  | * This code is distributed in the hope that it will be useful, but WITHOUT | 
|  | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | 
|  | * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License | 
|  | * version 2 for more details (a copy is included in the LICENSE file that | 
|  | * accompanied this code). | 
|  | * | 
|  | * You should have received a copy of the GNU General Public License version | 
|  | * 2 along with this work; if not, write to the Free Software Foundation, | 
|  | * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. | 
|  | * | 
|  | * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA | 
|  | * or visit www.oracle.com if you need additional information or have any | 
|  | * questions. | 
|  | */ | 
|  |  | 
|  | package java.util; | 
|  | import java.util.Map.Entry; | 
|  |  | 
|  | /** | 
|  | * This class provides a skeletal implementation of the <tt>Map</tt> | 
|  | * interface, to minimize the effort required to implement this interface. | 
|  | * | 
|  | * <p>To implement an unmodifiable map, the programmer needs only to extend this | 
|  | * class and provide an implementation for the <tt>entrySet</tt> method, which | 
|  | * returns a set-view of the map's mappings.  Typically, the returned set | 
|  | * will, in turn, be implemented atop <tt>AbstractSet</tt>.  This set should | 
|  | * not support the <tt>add</tt> or <tt>remove</tt> methods, and its iterator | 
|  | * should not support the <tt>remove</tt> method. | 
|  | * | 
|  | * <p>To implement a modifiable map, the programmer must additionally override | 
|  | * this class's <tt>put</tt> method (which otherwise throws an | 
|  | * <tt>UnsupportedOperationException</tt>), and the iterator returned by | 
|  | * <tt>entrySet().iterator()</tt> must additionally implement its | 
|  | * <tt>remove</tt> method. | 
|  | * | 
|  | * <p>The programmer should generally provide a void (no argument) and map | 
|  | * constructor, as per the recommendation in the <tt>Map</tt> interface | 
|  | * specification. | 
|  | * | 
|  | * <p>The documentation for each non-abstract method in this class describes its | 
|  | * implementation in detail.  Each of these methods may be overridden if the | 
|  | * map being implemented admits a more efficient implementation. | 
|  | * | 
|  | * <p>This class is a member of the | 
|  | * <a href="{@docRoot}/../technotes/guides/collections/index.html"> | 
|  | * Java Collections Framework</a>. | 
|  | * | 
|  | * @param <K> the type of keys maintained by this map | 
|  | * @param <V> the type of mapped values | 
|  | * | 
|  | * @author  Josh Bloch | 
|  | * @author  Neal Gafter | 
|  | * @see Map | 
|  | * @see Collection | 
|  | * @since 1.2 | 
|  | */ | 
|  |  | 
|  | public abstract class AbstractMap<K,V> implements Map<K,V> { | 
|  | /** | 
|  | * Sole constructor.  (For invocation by subclass constructors, typically | 
|  | * implicit.) | 
|  | */ | 
|  | protected AbstractMap() { | 
|  | } | 
|  |  | 
|  | // Query Operations | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation returns <tt>entrySet().size()</tt>. | 
|  | */ | 
|  | public int size() { | 
|  | return entrySet().size(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation returns <tt>size() == 0</tt>. | 
|  | */ | 
|  | public boolean isEmpty() { | 
|  | return size() == 0; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation iterates over <tt>entrySet()</tt> searching | 
|  | * for an entry with the specified value.  If such an entry is found, | 
|  | * <tt>true</tt> is returned.  If the iteration terminates without | 
|  | * finding such an entry, <tt>false</tt> is returned.  Note that this | 
|  | * implementation requires linear time in the size of the map. | 
|  | * | 
|  | * @throws ClassCastException   {@inheritDoc} | 
|  | * @throws NullPointerException {@inheritDoc} | 
|  | */ | 
|  | public boolean containsValue(Object value) { | 
|  | Iterator<Entry<K,V>> i = entrySet().iterator(); | 
|  | if (value==null) { | 
|  | while (i.hasNext()) { | 
|  | Entry<K,V> e = i.next(); | 
|  | if (e.getValue()==null) | 
|  | return true; | 
|  | } | 
|  | } else { | 
|  | while (i.hasNext()) { | 
|  | Entry<K,V> e = i.next(); | 
|  | if (value.equals(e.getValue())) | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation iterates over <tt>entrySet()</tt> searching | 
|  | * for an entry with the specified key.  If such an entry is found, | 
|  | * <tt>true</tt> is returned.  If the iteration terminates without | 
|  | * finding such an entry, <tt>false</tt> is returned.  Note that this | 
|  | * implementation requires linear time in the size of the map; many | 
|  | * implementations will override this method. | 
|  | * | 
|  | * @throws ClassCastException   {@inheritDoc} | 
|  | * @throws NullPointerException {@inheritDoc} | 
|  | */ | 
|  | public boolean containsKey(Object key) { | 
|  | Iterator<Map.Entry<K,V>> i = entrySet().iterator(); | 
|  | if (key==null) { | 
|  | while (i.hasNext()) { | 
|  | Entry<K,V> e = i.next(); | 
|  | if (e.getKey()==null) | 
|  | return true; | 
|  | } | 
|  | } else { | 
|  | while (i.hasNext()) { | 
|  | Entry<K,V> e = i.next(); | 
|  | if (key.equals(e.getKey())) | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation iterates over <tt>entrySet()</tt> searching | 
|  | * for an entry with the specified key.  If such an entry is found, | 
|  | * the entry's value is returned.  If the iteration terminates without | 
|  | * finding such an entry, <tt>null</tt> is returned.  Note that this | 
|  | * implementation requires linear time in the size of the map; many | 
|  | * implementations will override this method. | 
|  | * | 
|  | * @throws ClassCastException            {@inheritDoc} | 
|  | * @throws NullPointerException          {@inheritDoc} | 
|  | */ | 
|  | public V get(Object key) { | 
|  | Iterator<Entry<K,V>> i = entrySet().iterator(); | 
|  | if (key==null) { | 
|  | while (i.hasNext()) { | 
|  | Entry<K,V> e = i.next(); | 
|  | if (e.getKey()==null) | 
|  | return e.getValue(); | 
|  | } | 
|  | } else { | 
|  | while (i.hasNext()) { | 
|  | Entry<K,V> e = i.next(); | 
|  | if (key.equals(e.getKey())) | 
|  | return e.getValue(); | 
|  | } | 
|  | } | 
|  | return null; | 
|  | } | 
|  |  | 
|  |  | 
|  | // Modification Operations | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation always throws an | 
|  | * <tt>UnsupportedOperationException</tt>. | 
|  | * | 
|  | * @throws UnsupportedOperationException {@inheritDoc} | 
|  | * @throws ClassCastException            {@inheritDoc} | 
|  | * @throws NullPointerException          {@inheritDoc} | 
|  | * @throws IllegalArgumentException      {@inheritDoc} | 
|  | */ | 
|  | public V put(K key, V value) { | 
|  | throw new UnsupportedOperationException(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation iterates over <tt>entrySet()</tt> searching for an | 
|  | * entry with the specified key.  If such an entry is found, its value is | 
|  | * obtained with its <tt>getValue</tt> operation, the entry is removed | 
|  | * from the collection (and the backing map) with the iterator's | 
|  | * <tt>remove</tt> operation, and the saved value is returned.  If the | 
|  | * iteration terminates without finding such an entry, <tt>null</tt> is | 
|  | * returned.  Note that this implementation requires linear time in the | 
|  | * size of the map; many implementations will override this method. | 
|  | * | 
|  | * <p>Note that this implementation throws an | 
|  | * <tt>UnsupportedOperationException</tt> if the <tt>entrySet</tt> | 
|  | * iterator does not support the <tt>remove</tt> method and this map | 
|  | * contains a mapping for the specified key. | 
|  | * | 
|  | * @throws UnsupportedOperationException {@inheritDoc} | 
|  | * @throws ClassCastException            {@inheritDoc} | 
|  | * @throws NullPointerException          {@inheritDoc} | 
|  | */ | 
|  | public V remove(Object key) { | 
|  | Iterator<Entry<K,V>> i = entrySet().iterator(); | 
|  | Entry<K,V> correctEntry = null; | 
|  | if (key==null) { | 
|  | while (correctEntry==null && i.hasNext()) { | 
|  | Entry<K,V> e = i.next(); | 
|  | if (e.getKey()==null) | 
|  | correctEntry = e; | 
|  | } | 
|  | } else { | 
|  | while (correctEntry==null && i.hasNext()) { | 
|  | Entry<K,V> e = i.next(); | 
|  | if (key.equals(e.getKey())) | 
|  | correctEntry = e; | 
|  | } | 
|  | } | 
|  |  | 
|  | V oldValue = null; | 
|  | if (correctEntry !=null) { | 
|  | oldValue = correctEntry.getValue(); | 
|  | i.remove(); | 
|  | } | 
|  | return oldValue; | 
|  | } | 
|  |  | 
|  |  | 
|  | // Bulk Operations | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation iterates over the specified map's | 
|  | * <tt>entrySet()</tt> collection, and calls this map's <tt>put</tt> | 
|  | * operation once for each entry returned by the iteration. | 
|  | * | 
|  | * <p>Note that this implementation throws an | 
|  | * <tt>UnsupportedOperationException</tt> if this map does not support | 
|  | * the <tt>put</tt> operation and the specified map is nonempty. | 
|  | * | 
|  | * @throws UnsupportedOperationException {@inheritDoc} | 
|  | * @throws ClassCastException            {@inheritDoc} | 
|  | * @throws NullPointerException          {@inheritDoc} | 
|  | * @throws IllegalArgumentException      {@inheritDoc} | 
|  | */ | 
|  | public void putAll(Map<? extends K, ? extends V> m) { | 
|  | for (Map.Entry<? extends K, ? extends V> e : m.entrySet()) | 
|  | put(e.getKey(), e.getValue()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation calls <tt>entrySet().clear()</tt>. | 
|  | * | 
|  | * <p>Note that this implementation throws an | 
|  | * <tt>UnsupportedOperationException</tt> if the <tt>entrySet</tt> | 
|  | * does not support the <tt>clear</tt> operation. | 
|  | * | 
|  | * @throws UnsupportedOperationException {@inheritDoc} | 
|  | */ | 
|  | public void clear() { | 
|  | entrySet().clear(); | 
|  | } | 
|  |  | 
|  |  | 
|  | // Views | 
|  |  | 
|  | /** | 
|  | * Each of these fields are initialized to contain an instance of the | 
|  | * appropriate view the first time this view is requested.  The views are | 
|  | * stateless, so there's no reason to create more than one of each. | 
|  | * | 
|  | * <p>Since there is no synchronization performed while accessing these fields, | 
|  | * it is expected that java.util.Map view classes using these fields have | 
|  | * no non-final fields (or any fields at all except for outer-this). Adhering | 
|  | * to this rule would make the races on these fields benign. | 
|  | * | 
|  | * <p>It is also imperative that implementations read the field only once, | 
|  | * as in: | 
|  | * | 
|  | * <pre> {@code | 
|  | * public Set<K> keySet() { | 
|  | *   Set<K> ks = keySet;  // single racy read | 
|  | *   if (ks == null) { | 
|  | *     ks = new KeySet(); | 
|  | *     keySet = ks; | 
|  | *   } | 
|  | *   return ks; | 
|  | * } | 
|  | *}</pre> | 
|  | */ | 
|  | transient Set<K>        keySet; | 
|  | transient Collection<V> values; | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation returns a set that subclasses {@link AbstractSet}. | 
|  | * The subclass's iterator method returns a "wrapper object" over this | 
|  | * map's <tt>entrySet()</tt> iterator.  The <tt>size</tt> method | 
|  | * delegates to this map's <tt>size</tt> method and the | 
|  | * <tt>contains</tt> method delegates to this map's | 
|  | * <tt>containsKey</tt> method. | 
|  | * | 
|  | * <p>The set is created the first time this method is called, | 
|  | * and returned in response to all subsequent calls.  No synchronization | 
|  | * is performed, so there is a slight chance that multiple calls to this | 
|  | * method will not all return the same set. | 
|  | */ | 
|  | public Set<K> keySet() { | 
|  | Set<K> ks = keySet; | 
|  | if (ks == null) { | 
|  | ks = new AbstractSet<K>() { | 
|  | public Iterator<K> iterator() { | 
|  | return new Iterator<K>() { | 
|  | private Iterator<Entry<K,V>> i = entrySet().iterator(); | 
|  |  | 
|  | public boolean hasNext() { | 
|  | return i.hasNext(); | 
|  | } | 
|  |  | 
|  | public K next() { | 
|  | return i.next().getKey(); | 
|  | } | 
|  |  | 
|  | public void remove() { | 
|  | i.remove(); | 
|  | } | 
|  | }; | 
|  | } | 
|  |  | 
|  | public int size() { | 
|  | return AbstractMap.this.size(); | 
|  | } | 
|  |  | 
|  | public boolean isEmpty() { | 
|  | return AbstractMap.this.isEmpty(); | 
|  | } | 
|  |  | 
|  | public void clear() { | 
|  | AbstractMap.this.clear(); | 
|  | } | 
|  |  | 
|  | public boolean contains(Object k) { | 
|  | return AbstractMap.this.containsKey(k); | 
|  | } | 
|  | }; | 
|  | keySet = ks; | 
|  | } | 
|  | return ks; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * {@inheritDoc} | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation returns a collection that subclasses {@link | 
|  | * AbstractCollection}.  The subclass's iterator method returns a | 
|  | * "wrapper object" over this map's <tt>entrySet()</tt> iterator. | 
|  | * The <tt>size</tt> method delegates to this map's <tt>size</tt> | 
|  | * method and the <tt>contains</tt> method delegates to this map's | 
|  | * <tt>containsValue</tt> method. | 
|  | * | 
|  | * <p>The collection is created the first time this method is called, and | 
|  | * returned in response to all subsequent calls.  No synchronization is | 
|  | * performed, so there is a slight chance that multiple calls to this | 
|  | * method will not all return the same collection. | 
|  | */ | 
|  | public Collection<V> values() { | 
|  | Collection<V> vals = values; | 
|  | if (vals == null) { | 
|  | vals = new AbstractCollection<V>() { | 
|  | public Iterator<V> iterator() { | 
|  | return new Iterator<V>() { | 
|  | private Iterator<Entry<K,V>> i = entrySet().iterator(); | 
|  |  | 
|  | public boolean hasNext() { | 
|  | return i.hasNext(); | 
|  | } | 
|  |  | 
|  | public V next() { | 
|  | return i.next().getValue(); | 
|  | } | 
|  |  | 
|  | public void remove() { | 
|  | i.remove(); | 
|  | } | 
|  | }; | 
|  | } | 
|  |  | 
|  | public int size() { | 
|  | return AbstractMap.this.size(); | 
|  | } | 
|  |  | 
|  | public boolean isEmpty() { | 
|  | return AbstractMap.this.isEmpty(); | 
|  | } | 
|  |  | 
|  | public void clear() { | 
|  | AbstractMap.this.clear(); | 
|  | } | 
|  |  | 
|  | public boolean contains(Object v) { | 
|  | return AbstractMap.this.containsValue(v); | 
|  | } | 
|  | }; | 
|  | values = vals; | 
|  | } | 
|  | return vals; | 
|  | } | 
|  |  | 
|  | public abstract Set<Entry<K,V>> entrySet(); | 
|  |  | 
|  |  | 
|  | // Comparison and hashing | 
|  |  | 
|  | /** | 
|  | * Compares the specified object with this map for equality.  Returns | 
|  | * <tt>true</tt> if the given object is also a map and the two maps | 
|  | * represent the same mappings.  More formally, two maps <tt>m1</tt> and | 
|  | * <tt>m2</tt> represent the same mappings if | 
|  | * <tt>m1.entrySet().equals(m2.entrySet())</tt>.  This ensures that the | 
|  | * <tt>equals</tt> method works properly across different implementations | 
|  | * of the <tt>Map</tt> interface. | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation first checks if the specified object is this map; | 
|  | * if so it returns <tt>true</tt>.  Then, it checks if the specified | 
|  | * object is a map whose size is identical to the size of this map; if | 
|  | * not, it returns <tt>false</tt>.  If so, it iterates over this map's | 
|  | * <tt>entrySet</tt> collection, and checks that the specified map | 
|  | * contains each mapping that this map contains.  If the specified map | 
|  | * fails to contain such a mapping, <tt>false</tt> is returned.  If the | 
|  | * iteration completes, <tt>true</tt> is returned. | 
|  | * | 
|  | * @param o object to be compared for equality with this map | 
|  | * @return <tt>true</tt> if the specified object is equal to this map | 
|  | */ | 
|  | public boolean equals(Object o) { | 
|  | if (o == this) | 
|  | return true; | 
|  |  | 
|  | if (!(o instanceof Map)) | 
|  | return false; | 
|  | Map<?,?> m = (Map<?,?>) o; | 
|  | if (m.size() != size()) | 
|  | return false; | 
|  |  | 
|  | try { | 
|  | Iterator<Entry<K,V>> i = entrySet().iterator(); | 
|  | while (i.hasNext()) { | 
|  | Entry<K,V> e = i.next(); | 
|  | K key = e.getKey(); | 
|  | V value = e.getValue(); | 
|  | if (value == null) { | 
|  | if (!(m.get(key)==null && m.containsKey(key))) | 
|  | return false; | 
|  | } else { | 
|  | if (!value.equals(m.get(key))) | 
|  | return false; | 
|  | } | 
|  | } | 
|  | } catch (ClassCastException unused) { | 
|  | return false; | 
|  | } catch (NullPointerException unused) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the hash code value for this map.  The hash code of a map is | 
|  | * defined to be the sum of the hash codes of each entry in the map's | 
|  | * <tt>entrySet()</tt> view.  This ensures that <tt>m1.equals(m2)</tt> | 
|  | * implies that <tt>m1.hashCode()==m2.hashCode()</tt> for any two maps | 
|  | * <tt>m1</tt> and <tt>m2</tt>, as required by the general contract of | 
|  | * {@link Object#hashCode}. | 
|  | * | 
|  | * @implSpec | 
|  | * This implementation iterates over <tt>entrySet()</tt>, calling | 
|  | * {@link Map.Entry#hashCode hashCode()} on each element (entry) in the | 
|  | * set, and adding up the results. | 
|  | * | 
|  | * @return the hash code value for this map | 
|  | * @see Map.Entry#hashCode() | 
|  | * @see Object#equals(Object) | 
|  | * @see Set#equals(Object) | 
|  | */ | 
|  | public int hashCode() { | 
|  | int h = 0; | 
|  | Iterator<Entry<K,V>> i = entrySet().iterator(); | 
|  | while (i.hasNext()) | 
|  | h += i.next().hashCode(); | 
|  | return h; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a string representation of this map.  The string representation | 
|  | * consists of a list of key-value mappings in the order returned by the | 
|  | * map's <tt>entrySet</tt> view's iterator, enclosed in braces | 
|  | * (<tt>"{}"</tt>).  Adjacent mappings are separated by the characters | 
|  | * <tt>", "</tt> (comma and space).  Each key-value mapping is rendered as | 
|  | * the key followed by an equals sign (<tt>"="</tt>) followed by the | 
|  | * associated value.  Keys and values are converted to strings as by | 
|  | * {@link String#valueOf(Object)}. | 
|  | * | 
|  | * @return a string representation of this map | 
|  | */ | 
|  | public String toString() { | 
|  | Iterator<Entry<K,V>> i = entrySet().iterator(); | 
|  | if (! i.hasNext()) | 
|  | return "{}"; | 
|  |  | 
|  | StringBuilder sb = new StringBuilder(); | 
|  | sb.append('{'); | 
|  | for (;;) { | 
|  | Entry<K,V> e = i.next(); | 
|  | K key = e.getKey(); | 
|  | V value = e.getValue(); | 
|  | sb.append(key   == this ? "(this Map)" : key); | 
|  | sb.append('='); | 
|  | sb.append(value == this ? "(this Map)" : value); | 
|  | if (! i.hasNext()) | 
|  | return sb.append('}').toString(); | 
|  | sb.append(',').append(' '); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a shallow copy of this <tt>AbstractMap</tt> instance: the keys | 
|  | * and values themselves are not cloned. | 
|  | * | 
|  | * @return a shallow copy of this map | 
|  | */ | 
|  | protected Object clone() throws CloneNotSupportedException { | 
|  | AbstractMap<?,?> result = (AbstractMap<?,?>)super.clone(); | 
|  | result.keySet = null; | 
|  | result.values = null; | 
|  | return result; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Utility method for SimpleEntry and SimpleImmutableEntry. | 
|  | * Test for equality, checking for nulls. | 
|  | * | 
|  | * NB: Do not replace with Object.equals until JDK-8015417 is resolved. | 
|  | */ | 
|  | private static boolean eq(Object o1, Object o2) { | 
|  | return o1 == null ? o2 == null : o1.equals(o2); | 
|  | } | 
|  |  | 
|  | // Implementation Note: SimpleEntry and SimpleImmutableEntry | 
|  | // are distinct unrelated classes, even though they share | 
|  | // some code. Since you can't add or subtract final-ness | 
|  | // of a field in a subclass, they can't share representations, | 
|  | // and the amount of duplicated code is too small to warrant | 
|  | // exposing a common abstract class. | 
|  |  | 
|  |  | 
|  | /** | 
|  | * An Entry maintaining a key and a value.  The value may be | 
|  | * changed using the <tt>setValue</tt> method.  This class | 
|  | * facilitates the process of building custom map | 
|  | * implementations. For example, it may be convenient to return | 
|  | * arrays of <tt>SimpleEntry</tt> instances in method | 
|  | * <tt>Map.entrySet().toArray</tt>. | 
|  | * | 
|  | * @since 1.6 | 
|  | */ | 
|  | public static class SimpleEntry<K,V> | 
|  | implements Entry<K,V>, java.io.Serializable | 
|  | { | 
|  | private static final long serialVersionUID = -8499721149061103585L; | 
|  |  | 
|  | private final K key; | 
|  | private V value; | 
|  |  | 
|  | /** | 
|  | * Creates an entry representing a mapping from the specified | 
|  | * key to the specified value. | 
|  | * | 
|  | * @param key the key represented by this entry | 
|  | * @param value the value represented by this entry | 
|  | */ | 
|  | public SimpleEntry(K key, V value) { | 
|  | this.key   = key; | 
|  | this.value = value; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Creates an entry representing the same mapping as the | 
|  | * specified entry. | 
|  | * | 
|  | * @param entry the entry to copy | 
|  | */ | 
|  | public SimpleEntry(Entry<? extends K, ? extends V> entry) { | 
|  | this.key   = entry.getKey(); | 
|  | this.value = entry.getValue(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the key corresponding to this entry. | 
|  | * | 
|  | * @return the key corresponding to this entry | 
|  | */ | 
|  | public K getKey() { | 
|  | return key; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the value corresponding to this entry. | 
|  | * | 
|  | * @return the value corresponding to this entry | 
|  | */ | 
|  | public V getValue() { | 
|  | return value; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Replaces the value corresponding to this entry with the specified | 
|  | * value. | 
|  | * | 
|  | * @param value new value to be stored in this entry | 
|  | * @return the old value corresponding to the entry | 
|  | */ | 
|  | public V setValue(V value) { | 
|  | V oldValue = this.value; | 
|  | this.value = value; | 
|  | return oldValue; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Compares the specified object with this entry for equality. | 
|  | * Returns {@code true} if the given object is also a map entry and | 
|  | * the two entries represent the same mapping.  More formally, two | 
|  | * entries {@code e1} and {@code e2} represent the same mapping | 
|  | * if<pre> | 
|  | *   (e1.getKey()==null ? | 
|  | *    e2.getKey()==null : | 
|  | *    e1.getKey().equals(e2.getKey())) | 
|  | *   && | 
|  | *   (e1.getValue()==null ? | 
|  | *    e2.getValue()==null : | 
|  | *    e1.getValue().equals(e2.getValue()))</pre> | 
|  | * This ensures that the {@code equals} method works properly across | 
|  | * different implementations of the {@code Map.Entry} interface. | 
|  | * | 
|  | * @param o object to be compared for equality with this map entry | 
|  | * @return {@code true} if the specified object is equal to this map | 
|  | *         entry | 
|  | * @see    #hashCode | 
|  | */ | 
|  | public boolean equals(Object o) { | 
|  | if (!(o instanceof Map.Entry)) | 
|  | return false; | 
|  | Map.Entry<?,?> e = (Map.Entry<?,?>)o; | 
|  | return eq(key, e.getKey()) && eq(value, e.getValue()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the hash code value for this map entry.  The hash code | 
|  | * of a map entry {@code e} is defined to be: <pre> | 
|  | *   (e.getKey()==null   ? 0 : e.getKey().hashCode()) ^ | 
|  | *   (e.getValue()==null ? 0 : e.getValue().hashCode())</pre> | 
|  | * This ensures that {@code e1.equals(e2)} implies that | 
|  | * {@code e1.hashCode()==e2.hashCode()} for any two Entries | 
|  | * {@code e1} and {@code e2}, as required by the general | 
|  | * contract of {@link Object#hashCode}. | 
|  | * | 
|  | * @return the hash code value for this map entry | 
|  | * @see    #equals | 
|  | */ | 
|  | public int hashCode() { | 
|  | return (key   == null ? 0 :   key.hashCode()) ^ | 
|  | (value == null ? 0 : value.hashCode()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a String representation of this map entry.  This | 
|  | * implementation returns the string representation of this | 
|  | * entry's key followed by the equals character ("<tt>=</tt>") | 
|  | * followed by the string representation of this entry's value. | 
|  | * | 
|  | * @return a String representation of this map entry | 
|  | */ | 
|  | public String toString() { | 
|  | return key + "=" + value; | 
|  | } | 
|  |  | 
|  | } | 
|  |  | 
|  | /** | 
|  | * An Entry maintaining an immutable key and value.  This class | 
|  | * does not support method <tt>setValue</tt>.  This class may be | 
|  | * convenient in methods that return thread-safe snapshots of | 
|  | * key-value mappings. | 
|  | * | 
|  | * @since 1.6 | 
|  | */ | 
|  | public static class SimpleImmutableEntry<K,V> | 
|  | implements Entry<K,V>, java.io.Serializable | 
|  | { | 
|  | private static final long serialVersionUID = 7138329143949025153L; | 
|  |  | 
|  | private final K key; | 
|  | private final V value; | 
|  |  | 
|  | /** | 
|  | * Creates an entry representing a mapping from the specified | 
|  | * key to the specified value. | 
|  | * | 
|  | * @param key the key represented by this entry | 
|  | * @param value the value represented by this entry | 
|  | */ | 
|  | public SimpleImmutableEntry(K key, V value) { | 
|  | this.key   = key; | 
|  | this.value = value; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Creates an entry representing the same mapping as the | 
|  | * specified entry. | 
|  | * | 
|  | * @param entry the entry to copy | 
|  | */ | 
|  | public SimpleImmutableEntry(Entry<? extends K, ? extends V> entry) { | 
|  | this.key   = entry.getKey(); | 
|  | this.value = entry.getValue(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the key corresponding to this entry. | 
|  | * | 
|  | * @return the key corresponding to this entry | 
|  | */ | 
|  | public K getKey() { | 
|  | return key; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the value corresponding to this entry. | 
|  | * | 
|  | * @return the value corresponding to this entry | 
|  | */ | 
|  | public V getValue() { | 
|  | return value; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Replaces the value corresponding to this entry with the specified | 
|  | * value (optional operation).  This implementation simply throws | 
|  | * <tt>UnsupportedOperationException</tt>, as this class implements | 
|  | * an <i>immutable</i> map entry. | 
|  | * | 
|  | * @param value new value to be stored in this entry | 
|  | * @return (Does not return) | 
|  | * @throws UnsupportedOperationException always | 
|  | */ | 
|  | public V setValue(V value) { | 
|  | throw new UnsupportedOperationException(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Compares the specified object with this entry for equality. | 
|  | * Returns {@code true} if the given object is also a map entry and | 
|  | * the two entries represent the same mapping.  More formally, two | 
|  | * entries {@code e1} and {@code e2} represent the same mapping | 
|  | * if<pre> | 
|  | *   (e1.getKey()==null ? | 
|  | *    e2.getKey()==null : | 
|  | *    e1.getKey().equals(e2.getKey())) | 
|  | *   && | 
|  | *   (e1.getValue()==null ? | 
|  | *    e2.getValue()==null : | 
|  | *    e1.getValue().equals(e2.getValue()))</pre> | 
|  | * This ensures that the {@code equals} method works properly across | 
|  | * different implementations of the {@code Map.Entry} interface. | 
|  | * | 
|  | * @param o object to be compared for equality with this map entry | 
|  | * @return {@code true} if the specified object is equal to this map | 
|  | *         entry | 
|  | * @see    #hashCode | 
|  | */ | 
|  | public boolean equals(Object o) { | 
|  | if (!(o instanceof Map.Entry)) | 
|  | return false; | 
|  | Map.Entry<?,?> e = (Map.Entry<?,?>)o; | 
|  | return eq(key, e.getKey()) && eq(value, e.getValue()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the hash code value for this map entry.  The hash code | 
|  | * of a map entry {@code e} is defined to be: <pre> | 
|  | *   (e.getKey()==null   ? 0 : e.getKey().hashCode()) ^ | 
|  | *   (e.getValue()==null ? 0 : e.getValue().hashCode())</pre> | 
|  | * This ensures that {@code e1.equals(e2)} implies that | 
|  | * {@code e1.hashCode()==e2.hashCode()} for any two Entries | 
|  | * {@code e1} and {@code e2}, as required by the general | 
|  | * contract of {@link Object#hashCode}. | 
|  | * | 
|  | * @return the hash code value for this map entry | 
|  | * @see    #equals | 
|  | */ | 
|  | public int hashCode() { | 
|  | return (key   == null ? 0 :   key.hashCode()) ^ | 
|  | (value == null ? 0 : value.hashCode()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a String representation of this map entry.  This | 
|  | * implementation returns the string representation of this | 
|  | * entry's key followed by the equals character ("<tt>=</tt>") | 
|  | * followed by the string representation of this entry's value. | 
|  | * | 
|  | * @return a String representation of this map entry | 
|  | */ | 
|  | public String toString() { | 
|  | return key + "=" + value; | 
|  | } | 
|  |  | 
|  | } | 
|  |  | 
|  | } |