View Javadoc
1   /*
2    * Licensed to the Apache Software Foundation (ASF) under one or more
3    * contributor license agreements.  See the NOTICE file distributed with
4    * this work for additional information regarding copyright ownership.
5    * The ASF licenses this file to You under the Apache License, Version 2.0
6    * (the "License"); you may not use this file except in compliance with
7    * the License.  You may obtain a copy of the License at
8    *
9    *      http://www.apache.org/licenses/LICENSE-2.0
10   *
11   * Unless required by applicable law or agreed to in writing, software
12   * distributed under the License is distributed on an "AS IS" BASIS,
13   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14   * See the License for the specific language governing permissions and
15   * limitations under the License.
16   */
17  package org.apache.commons.collections4.map;
18  
19  import java.io.Serializable;
20  import java.util.Arrays;
21  import java.util.Collection;
22  import java.util.Map;
23  import java.util.Set;
24  
25  import org.apache.commons.collections4.CollectionUtils;
26  import org.apache.commons.collections4.collection.CompositeCollection;
27  import org.apache.commons.collections4.set.CompositeSet;
28  
29  /**
30   * Decorates a map of other maps to provide a single unified view.
31   * <p>
32   * Changes made to this map will actually be made on the decorated map.
33   * Add and remove operations require the use of a pluggable strategy. If no
34   * strategy is provided then add and remove are unsupported.
35   * </p>
36   * <p>
37   * <strong>Note that CompositeMap is not synchronized and is not thread-safe.</strong>
38   * If you wish to use this map from multiple threads concurrently, you must use
39   * appropriate synchronization. The simplest approach is to wrap this map
40   * using {@link java.util.Collections#synchronizedMap(Map)}. This class may throw
41   * exceptions when accessed by concurrent threads without synchronization.
42   * </p>
43   *
44   * @param <K> the type of the keys in this map
45   * @param <V> the type of the values in this map
46   * @since 3.0
47   */
48  public class CompositeMap<K, V> extends AbstractIterableMap<K, V> implements Serializable {
49  
50      /**
51       * This interface allows definition for all of the indeterminate
52       * mutators in a CompositeMap, as well as providing a hook for
53       * callbacks on key collisions.
54       *
55       * @param <K> the type of the keys in the map
56       * @param <V> the type of the values in the map
57       */
58      public interface MapMutator<K, V> extends Serializable {
59          /**
60           * Called when the CompositeMap.put() method is invoked.
61           *
62           * @param map  the CompositeMap which is being modified
63           * @param composited  array of Maps in the CompositeMap being modified
64           * @param key  key with which the specified value is to be associated.
65           * @param value  value to be associated with the specified key.
66           * @return previous value associated with specified key, or {@code null}
67           *         if there was no mapping for key.  A {@code null} return can
68           *         also indicate that the map previously associated {@code null}
69           *         with the specified key, if the implementation supports
70           *         {@code null} values.
71           *
72           * @throws UnsupportedOperationException if not defined
73           * @throws ClassCastException if the class of the specified key or value
74           *            prevents it from being stored in this map.
75           * @throws IllegalArgumentException if some aspect of this key or value
76           *            prevents it from being stored in this map.
77           * @throws NullPointerException this map does not permit {@code null}
78           *            keys or values, and the specified key or value is
79           *            {@code null}.
80           */
81          V put(CompositeMap<K, V> map, Map<K, V>[] composited, K key, V value);
82  
83          /**
84           * Called when the CompositeMap.putAll() method is invoked.
85           *
86           * @param map  the CompositeMap which is being modified
87           * @param composited  array of Maps in the CompositeMap being modified
88           * @param mapToAdd  Mappings to be stored in this CompositeMap
89           * @throws UnsupportedOperationException if not defined
90           * @throws ClassCastException if the class of the specified key or value
91           *            prevents it from being stored in this map.
92           * @throws IllegalArgumentException if some aspect of this key or value
93           *            prevents it from being stored in this map.
94           * @throws NullPointerException this map does not permit {@code null}
95           *            keys or values, and the specified key or value is
96           *            {@code null}.
97           */
98          void putAll(CompositeMap<K, V> map, Map<K, V>[] composited,
99                  Map<? extends K, ? extends V> mapToAdd);
100 
101         /**
102          * Called when adding a new Composited Map results in a
103          * key collision.
104          *
105          * @param composite  the CompositeMap with the collision
106          * @param existing  the Map already in the composite which contains the
107          *        offending key
108          * @param added  the Map being added
109          * @param intersect  the intersection of the keysets of the existing and added maps
110          */
111         void resolveCollision(CompositeMap<K, V> composite, Map<K, V> existing,
112                 Map<K, V> added, Collection<K> intersect);
113     }
114 
115     @SuppressWarnings("rawtypes")
116     private static final Map[] EMPTY_MAP_ARRAY = {};
117 
118     /** Serialization version */
119     private static final long serialVersionUID = -6096931280583808322L;
120 
121     /** Array of all maps in the composite */
122     private Map<K, V>[] composite;
123 
124     /** Handle mutation operations */
125     private MapMutator<K, V> mutator;
126 
127     /**
128      * Create a new, empty, CompositeMap.
129      */
130     @SuppressWarnings("unchecked")
131     public CompositeMap() {
132         this(new Map[] {}, null);
133     }
134 
135     /**
136      * Create a new CompositeMap which composites all of the Map instances in the
137      * argument. It copies the argument array, it does not use it directly.
138      *
139      * @param composite  the Maps to be composited
140      * @throws IllegalArgumentException if there is a key collision
141      */
142     public CompositeMap(final Map<K, V>... composite) {
143         this(composite, null);
144     }
145 
146     /**
147      * Create a new CompositeMap with two composited Map instances.
148      *
149      * @param one  the first Map to be composited
150      * @param two  the second Map to be composited
151      * @throws IllegalArgumentException if there is a key collision
152      */
153     @SuppressWarnings("unchecked")
154     public CompositeMap(final Map<K, V> one, final Map<K, V> two) {
155         this(new Map[] { one, two }, null);
156     }
157 
158     /**
159      * Create a new CompositeMap with two composited Map instances.
160      *
161      * @param one  the first Map to be composited
162      * @param two  the second Map to be composited
163      * @param mutator  MapMutator to be used for mutation operations
164      */
165     @SuppressWarnings("unchecked")
166     public CompositeMap(final Map<K, V> one, final Map<K, V> two, final MapMutator<K, V> mutator) {
167         this(new Map[] { one, two }, mutator);
168     }
169 
170     /**
171      * Create a new CompositeMap which composites all of the Map instances in the
172      * argument. It copies the argument array, it does not use it directly.
173      *
174      * @param composite  Maps to be composited
175      * @param mutator  MapMutator to be used for mutation operations
176      */
177     @SuppressWarnings("unchecked")
178     public CompositeMap(final Map<K, V>[] composite, final MapMutator<K, V> mutator) {
179         this.mutator = mutator;
180         this.composite = EMPTY_MAP_ARRAY;
181         for (int i = composite.length - 1; i >= 0; --i) {
182             this.addComposited(composite[i]);
183         }
184     }
185 
186     /**
187      * Add an additional Map to the composite.
188      *
189      * @param map  the Map to be added to the composite
190      * @throws IllegalArgumentException if there is a key collision and there is no
191      *         MapMutator set to handle it.
192      */
193     public synchronized void addComposited(final Map<K, V> map) throws IllegalArgumentException {
194         if (map != null) {
195             for (int i = composite.length - 1; i >= 0; --i) {
196                 final Collection<K> intersect = CollectionUtils.intersection(composite[i].keySet(), map.keySet());
197                 if (!intersect.isEmpty()) {
198                     if (mutator == null) {
199                         throw new IllegalArgumentException("Key collision adding Map to CompositeMap");
200                     }
201                     mutator.resolveCollision(this, composite[i], map, intersect);
202                 }
203             }
204             final Map<K, V>[] temp = Arrays.copyOf(composite, composite.length + 1);
205             temp[temp.length - 1] = map;
206             composite = temp;
207         }
208     }
209 
210     /**
211      * Calls {@code clear()} on all composited Maps.
212      *
213      * @throws UnsupportedOperationException if any of the composited Maps do not support clear()
214      */
215     @Override
216     public void clear() {
217         for (int i = composite.length - 1; i >= 0; --i) {
218             composite[i].clear();
219         }
220     }
221 
222     /**
223      * Returns {@code true} if this map contains a mapping for the specified
224      * key.  More formally, returns {@code true} if and only if
225      * this map contains at a mapping for a key {@code k} such that
226      * {@code (key==null ? k==null : key.equals(k))}.  (There can be
227      * at most one such mapping.)
228      *
229      * @param key  key whose presence in this map is to be tested.
230      * @return {@code true} if this map contains a mapping for the specified
231      *         key.
232      *
233      * @throws ClassCastException if the key is of an inappropriate type for
234      *         this map (optional).
235      * @throws NullPointerException if the key is {@code null} and this map
236      *            does not permit {@code null} keys (optional).
237      */
238     @Override
239     public boolean containsKey(final Object key) {
240         for (int i = composite.length - 1; i >= 0; --i) {
241             if (composite[i].containsKey(key)) {
242                 return true;
243             }
244         }
245         return false;
246     }
247 
248     /**
249      * Returns {@code true} if this map maps one or more keys to the
250      * specified value.  More formally, returns {@code true} if and only if
251      * this map contains at least one mapping to a value {@code v} such that
252      * {@code (value==null ? v==null : value.equals(v))}.  This operation
253      * will probably require time linear in the map size for most
254      * implementations of the {@code Map} interface.
255      *
256      * @param value value whose presence in this map is to be tested.
257      * @return {@code true} if this map maps one or more keys to the
258      *         specified value.
259      * @throws ClassCastException if the value is of an inappropriate type for
260      *         this map (optional).
261      * @throws NullPointerException if the value is {@code null} and this map
262      *            does not permit {@code null} values (optional).
263      */
264     @Override
265     public boolean containsValue(final Object value) {
266         for (int i = composite.length - 1; i >= 0; --i) {
267             if (composite[i].containsValue(value)) {
268                 return true;
269             }
270         }
271         return false;
272     }
273 
274     /**
275      * Returns a set view of the mappings contained in this map.  Each element
276      * in the returned set is a {@code Map.Entry}.  The set is backed by the
277      * map, so changes to the map are reflected in the set, and vice-versa.
278      * If the map is modified while an iteration over the set is in progress,
279      * the results of the iteration are undefined.  The set supports element
280      * removal, which removes the corresponding mapping from the map, via the
281      * {@code Iterator.remove}, {@code Set.remove}, {@code removeAll},
282      * {@code retainAll} and {@code clear} operations.  It does not support
283      * the {@code add} or {@code addAll} operations.
284      * <p>
285      * This implementation returns a {@code CompositeSet} which
286      * composites the entry sets from all of the composited maps.
287      *
288      * @see CompositeSet
289      * @return a set view of the mappings contained in this map.
290      */
291     @Override
292     public Set<Map.Entry<K, V>> entrySet() {
293         final CompositeSet<Map.Entry<K, V>> entries = new CompositeSet<>();
294         for (int i = composite.length - 1; i >= 0; --i) {
295             entries.addComposited(composite[i].entrySet());
296         }
297         return entries;
298     }
299 
300     /**
301      * Checks if this Map equals another as per the Map specification.
302      *
303      * @param obj  the object to compare to
304      * @return true if the maps are equal
305      */
306     @Override
307     public boolean equals(final Object obj) {
308         if (obj instanceof Map) {
309             final Map<?, ?> map = (Map<?, ?>) obj;
310             return this.entrySet().equals(map.entrySet());
311         }
312         return false;
313     }
314 
315     /**
316      * Returns the value to which this map maps the specified key.  Returns
317      * {@code null} if the map contains no mapping for this key.  A return
318      * value of {@code null} does not <em>necessarily</em> indicate that the
319      * map contains no mapping for the key; it's also possible that the map
320      * explicitly maps the key to {@code null}.  The {@code containsKey}
321      * operation may be used to distinguish these two cases.
322      *
323      * <p>More formally, if this map contains a mapping from a key
324      * {@code k} to a value {@code v} such that {@code (key==null ? k==null :
325      * key.equals(k))}, then this method returns {@code v}; otherwise
326      * it returns {@code null}.  (There can be at most one such mapping.)
327      *
328      * @param key key whose associated value is to be returned.
329      * @return the value to which this map maps the specified key, or
330      *         {@code null} if the map contains no mapping for this key.
331      *
332      * @throws ClassCastException if the key is of an inappropriate type for
333      *         this map (optional).
334      * @throws NullPointerException key is {@code null} and this map does
335      *         not permit {@code null} keys (optional).
336      *
337      * @see #containsKey(Object)
338      */
339     @Override
340     public V get(final Object key) {
341         for (int i = composite.length - 1; i >= 0; --i) {
342             if (composite[i].containsKey(key)) {
343                 return composite[i].get(key);
344             }
345         }
346         return null;
347     }
348 
349     /**
350      * Gets a hash code for the Map as per the Map specification.
351      * {@inheritDoc}
352      */
353     @Override
354     public int hashCode() {
355         int code = 0;
356         for (final Map.Entry<K, V> entry : entrySet()) {
357             code += entry.hashCode();
358         }
359         return code;
360     }
361 
362     /**
363      * Returns {@code true} if this map contains no key-value mappings.
364      *
365      * @return {@code true} if this map contains no key-value mappings.
366      */
367     @Override
368     public boolean isEmpty() {
369         for (int i = composite.length - 1; i >= 0; --i) {
370             if (!composite[i].isEmpty()) {
371                 return false;
372             }
373         }
374         return true;
375     }
376 
377     /**
378      * Returns a set view of the keys contained in this map.  The set is
379      * backed by the map, so changes to the map are reflected in the set, and
380      * vice-versa.  If the map is modified while an iteration over the set is
381      * in progress, the results of the iteration are undefined.  The set
382      * supports element removal, which removes the corresponding mapping from
383      * the map, via the {@code Iterator.remove}, {@code Set.remove},
384      * {@code removeAll} {@code retainAll}, and {@code clear} operations.
385      * It does not support the add or {@code addAll} operations.
386      * <p>
387      * This implementation returns a {@code CompositeSet} which
388      * composites the key sets from all of the composited maps.
389      * </p>
390      *
391      * @return a set view of the keys contained in this map.
392      */
393     @Override
394     public Set<K> keySet() {
395         final CompositeSet<K> keys = new CompositeSet<>();
396         for (int i = composite.length - 1; i >= 0; --i) {
397             keys.addComposited(composite[i].keySet());
398         }
399         return keys;
400     }
401 
402     /**
403      * Associates the specified value with the specified key in this map
404      * (optional operation).  If the map previously contained a mapping for
405      * this key, the old value is replaced by the specified value.  (A map
406      * {@code m} is said to contain a mapping for a key {@code k} if and only
407      * if {@link #containsKey(Object) m.containsKey(k)} would return
408      * {@code true}.))
409      *
410      * @param key key with which the specified value is to be associated.
411      * @param value value to be associated with the specified key.
412      * @return previous value associated with specified key, or {@code null}
413      *         if there was no mapping for key.  A {@code null} return can
414      *         also indicate that the map previously associated {@code null}
415      *         with the specified key, if the implementation supports
416      *         {@code null} values.
417      *
418      * @throws UnsupportedOperationException if no MapMutator has been specified
419      * @throws ClassCastException if the class of the specified key or value
420      *            prevents it from being stored in this map.
421      * @throws IllegalArgumentException if some aspect of this key or value
422      *            prevents it from being stored in this map.
423      * @throws NullPointerException this map does not permit {@code null}
424      *            keys or values, and the specified key or value is
425      *            {@code null}.
426      */
427     @Override
428     public V put(final K key, final V value) {
429         if (mutator == null) {
430             throw new UnsupportedOperationException("No mutator specified");
431         }
432         return mutator.put(this, composite, key, value);
433     }
434 
435     /**
436      * Copies all of the mappings from the specified map to this map
437      * (optional operation).  The effect of this call is equivalent to that
438      * of calling {@link #put(Object,Object) put(k, v)} on this map once
439      * for each mapping from key {@code k} to value {@code v} in the
440      * specified map.  The behavior of this operation is unspecified if the
441      * specified map is modified while the operation is in progress.
442      *
443      * @param map Mappings to be stored in this map.
444      * @throws UnsupportedOperationException if the {@code putAll} method is
445      *         not supported by this map.
446      *
447      * @throws ClassCastException if the class of a key or value in the
448      *         specified map prevents it from being stored in this map.
449      *
450      * @throws IllegalArgumentException some aspect of a key or value in the
451      *         specified map prevents it from being stored in this map.
452      * @throws NullPointerException the specified map is {@code null}, or if
453      *         this map does not permit {@code null} keys or values, and the
454      *         specified map contains {@code null} keys or values.
455      */
456     @Override
457     public void putAll(final Map<? extends K, ? extends V> map) {
458         if (mutator == null) {
459             throw new UnsupportedOperationException("No mutator specified");
460         }
461         mutator.putAll(this, composite, map);
462     }
463 
464     /**
465      * Removes the mapping for this key from this map if it is present
466      * (optional operation).   More formally, if this map contains a mapping
467      * from key {@code k} to value {@code v} such that
468      * {@code (key==null ?  k==null : key.equals(k))}, that mapping
469      * is removed.  (The map can contain at most one such mapping.)
470      *
471      * <p>Returns the value to which the map previously associated the key, or
472      * {@code null} if the map contained no mapping for this key.  (A
473      * {@code null} return can also indicate that the map previously
474      * associated {@code null} with the specified key if the implementation
475      * supports {@code null} values.)  The map will not contain a mapping for
476      * the specified  key once the call returns.
477      *
478      * @param key key whose mapping is to be removed from the map.
479      * @return previous value associated with specified key, or {@code null}
480      *         if there was no mapping for key.
481      *
482      * @throws ClassCastException if the key is of an inappropriate type for
483      *         the composited map (optional).
484      * @throws NullPointerException if the key is {@code null} and the composited map
485      *            does not permit {@code null} keys (optional).
486      * @throws UnsupportedOperationException if the {@code remove} method is
487      *         not supported by the composited map containing the key
488      */
489     @Override
490     public V remove(final Object key) {
491         for (int i = composite.length - 1; i >= 0; --i) {
492             if (composite[i].containsKey(key)) {
493                 return composite[i].remove(key);
494             }
495         }
496         return null;
497     }
498 
499     /**
500      * Remove a Map from the composite.
501      *
502      * @param map  the Map to be removed from the composite
503      * @return The removed Map or {@code null} if map is not in the composite
504      */
505     @SuppressWarnings("unchecked")
506     public synchronized Map<K, V> removeComposited(final Map<K, V> map) {
507         final int size = composite.length;
508         for (int i = 0; i < size; ++i) {
509             if (composite[i].equals(map)) {
510                 final Map<K, V>[] temp = new Map[size - 1];
511                 System.arraycopy(composite, 0, temp, 0, i);
512                 System.arraycopy(composite, i + 1, temp, i, size - i - 1);
513                 composite = temp;
514                 return map;
515             }
516         }
517         return null;
518     }
519 
520     /**
521      * Specify the MapMutator to be used by mutation operations.
522      *
523      * @param mutator  the MapMutator to be used for mutation delegation
524      */
525     public void setMutator(final MapMutator<K, V> mutator) {
526         this.mutator = mutator;
527     }
528 
529     /**
530      * Returns the number of key-value mappings in this map.  If the
531      * map contains more than {@code Integer.MAX_VALUE} elements, returns
532      * {@code Integer.MAX_VALUE}.
533      *
534      * @return the number of key-value mappings in this map.
535      */
536     @Override
537     public int size() {
538         int size = 0;
539         for (int i = composite.length - 1; i >= 0; --i) {
540             size += composite[i].size();
541         }
542         return size;
543     }
544 
545     /**
546      * Returns a collection view of the values contained in this map.  The
547      * collection is backed by the map, so changes to the map are reflected in
548      * the collection, and vice-versa.  If the map is modified while an
549      * iteration over the collection is in progress, the results of the
550      * iteration are undefined.  The collection supports element removal,
551      * which removes the corresponding mapping from the map, via the
552      * {@code Iterator.remove}, {@code Collection.remove},
553      * {@code removeAll}, {@code retainAll} and {@code clear} operations.
554      * It does not support the add or {@code addAll} operations.
555      *
556      * @return a collection view of the values contained in this map.
557      */
558     @Override
559     public Collection<V> values() {
560         final CompositeCollection<V> values = new CompositeCollection<>();
561         for (int i = composite.length - 1; i >= 0; --i) {
562             values.addComposited(composite[i].values());
563         }
564         return values;
565     }
566 }