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.multimap;
18  
19  import java.io.IOException;
20  import java.io.ObjectInputStream;
21  import java.io.ObjectOutputStream;
22  import java.io.Serializable;
23  import java.util.ArrayList;
24  import java.util.Collection;
25  import java.util.LinkedHashMap;
26  import java.util.Map;
27  
28  import org.apache.commons.collections4.MultiValuedMap;
29  
30  /**
31   * Implements a {@code ListValuedMap}, using a {@link LinkedHashMap} to provide data
32   * storage and {@link ArrayList}s as value collections. This is the standard
33   * implementation of a ListValuedMap.
34   * <p>
35   * <strong>Note that ArrayListValuedLinkedHashMap is not synchronized and is not
36   * thread-safe.</strong> If you wish to use this map from multiple threads
37   * concurrently, you must use appropriate synchronization. This class may throw
38   * exceptions when accessed by concurrent threads without synchronization.
39   * </p>
40   *
41   * @param <K> the type of the keys in this map
42   * @param <V> the type of the values in this map
43   * @since 4.5.0-M3
44   */
45  public class ArrayListValuedLinkedHashMap<K, V> extends AbstractListValuedMap<K, V>
46      implements Serializable {
47  
48      /** Serialization Version */
49      private static final long serialVersionUID = 20241014L;
50  
51      /**
52       * The initial map capacity used when none specified in constructor.
53       */
54      private static final int DEFAULT_INITIAL_MAP_CAPACITY = 16;
55  
56      /**
57       * The initial list capacity when using none specified in constructor.
58       */
59      private static final int DEFAULT_INITIAL_LIST_CAPACITY = 3;
60  
61      /**
62       * The initial list capacity when creating a new value collection.
63       */
64      private final int initialListCapacity;
65  
66      /**
67       * Creates an empty ArrayListValuedHashMap with the default initial
68       * map capacity (16) and the default initial list capacity (3).
69       */
70      public ArrayListValuedLinkedHashMap() {
71          this(DEFAULT_INITIAL_MAP_CAPACITY, DEFAULT_INITIAL_LIST_CAPACITY);
72      }
73  
74      /**
75       * Creates an empty ArrayListValuedHashMap with the default initial
76       * map capacity (16) and the specified initial list capacity.
77       *
78       * @param initialListCapacity  the initial capacity used for value collections
79       */
80      public ArrayListValuedLinkedHashMap(final int initialListCapacity) {
81          this(DEFAULT_INITIAL_MAP_CAPACITY, initialListCapacity);
82      }
83  
84      /**
85       * Creates an empty ArrayListValuedHashMap with the specified initial
86       * map and list capacities.
87       *
88       * @param initialMapCapacity  the initial hashmap capacity
89       * @param initialListCapacity  the initial capacity used for value collections
90       */
91      public ArrayListValuedLinkedHashMap(final int initialMapCapacity, final int initialListCapacity) {
92          super(new LinkedHashMap<>(initialMapCapacity));
93          this.initialListCapacity = initialListCapacity;
94      }
95  
96      /**
97       * Creates an ArrayListValuedHashMap copying all the mappings of the given map.
98       *
99       * @param map a {@code Map} to copy into this map
100      */
101     public ArrayListValuedLinkedHashMap(final Map<? extends K, ? extends V> map) {
102         this(map.size(), DEFAULT_INITIAL_LIST_CAPACITY);
103         super.putAll(map);
104     }
105 
106     /**
107      * Creates an ArrayListValuedHashMap copying all the mappings of the given map.
108      *
109      * @param map a {@code MultiValuedMap} to copy into this map
110      */
111     public ArrayListValuedLinkedHashMap(final MultiValuedMap<? extends K, ? extends V> map) {
112         this(map.size(), DEFAULT_INITIAL_LIST_CAPACITY);
113         super.putAll(map);
114     }
115 
116     @Override
117     protected ArrayList<V> createCollection() {
118         return new ArrayList<>(initialListCapacity);
119     }
120 
121     /**
122      * Deserializes an instance from an ObjectInputStream.
123      *
124      * @param in The source ObjectInputStream.
125      * @throws IOException            Any of the usual Input/Output related exceptions.
126      * @throws ClassNotFoundException A class of a serialized object cannot be found.
127      */
128     private void readObject(final ObjectInputStream in) throws IOException, ClassNotFoundException {
129         in.defaultReadObject();
130         setMap(new LinkedHashMap<>());
131         doReadObject(in);
132     }
133 
134     /**
135      * Trims the capacity of all value collections to their current size.
136      */
137     public void trimToSize() {
138         for (final Collection<V> coll : getMap().values()) {
139             final ArrayList<V> list = (ArrayList<V>) coll;
140             list.trimToSize();
141         }
142     }
143 
144     /**
145      * Serializes this object to an ObjectOutputStream.
146      *
147      * @param out the target ObjectOutputStream.
148      * @throws IOException thrown when an I/O errors occur writing to the target stream.
149      */
150     private void writeObject(final ObjectOutputStream out) throws IOException {
151         out.defaultWriteObject();
152         doWriteObject(out);
153     }
154 
155 }