001/*
002 * Copyright (C) 2013 The Guava Authors
003 *
004 * Licensed under the Apache License, Version 2.0 (the "License");
005 * you may not use this file except in compliance with the License.
006 * You may obtain a copy of the License at
007 *
008 * http://www.apache.org/licenses/LICENSE-2.0
009 *
010 * Unless required by applicable law or agreed to in writing, software
011 * distributed under the License is distributed on an "AS IS" BASIS,
012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 * See the License for the specific language governing permissions and
014 * limitations under the License.
015 */
016
017package com.google.common.collect;
018
019import static com.google.common.base.Preconditions.checkNotNull;
020import static com.google.common.collect.CollectPreconditions.checkNonnegative;
021import static com.google.common.collect.Maps.newLinkedHashMapWithExpectedSize;
022
023import com.google.common.annotations.Beta;
024import com.google.common.annotations.GwtCompatible;
025import com.google.common.base.Supplier;
026import java.io.Serializable;
027import java.util.ArrayList;
028import java.util.Collection;
029import java.util.Comparator;
030import java.util.EnumMap;
031import java.util.EnumSet;
032import java.util.HashMap;
033import java.util.HashSet;
034import java.util.LinkedHashMap;
035import java.util.LinkedHashSet;
036import java.util.LinkedList;
037import java.util.List;
038import java.util.Map;
039import java.util.Set;
040import java.util.SortedSet;
041import java.util.TreeMap;
042import java.util.TreeSet;
043
044/**
045 * A builder for a multimap implementation that allows customization of the backing map and value
046 * collection implementations used in a particular multimap.
047 *
048 * <p>This can be used to easily configure multimap data structure implementations not provided
049 * explicitly in {@code com.google.common.collect}, for example:
050 *
051 * <pre>   {@code
052 *   ListMultimap<String, Integer> treeListMultimap =
053 *       MultimapBuilder.treeKeys().arrayListValues().build();
054 *   SetMultimap<Integer, MyEnum> hashEnumMultimap =
055 *       MultimapBuilder.hashKeys().enumSetValues(MyEnum.class).build();}</pre>
056 *
057 * <p>{@code MultimapBuilder} instances are immutable.  Invoking a configuration method has no
058 * effect on the receiving instance; you must store and use the new builder instance it returns
059 * instead.
060 *
061 * <p>The generated multimaps are serializable if the key and value types are serializable,
062 * unless stated otherwise in one of the configuration methods.
063 *
064 * @author Louis Wasserman
065 * @param <K0> An upper bound on the key type of the generated multimap.
066 * @param <V0> An upper bound on the value type of the generated multimap.
067 * @since 16.0
068 */
069@Beta
070@GwtCompatible
071public abstract class MultimapBuilder<K0, V0> {
072  /*
073   * Leaving K and V as upper bounds rather than the actual key and value types allows type
074   * parameters to be left implicit more often. CacheBuilder uses the same technique.
075   */
076
077  private MultimapBuilder() {}
078
079  private static final int DEFAULT_EXPECTED_KEYS = 8;
080
081  /**
082   * Uses a {@link HashMap} to map keys to value collections.
083   */
084  public static MultimapBuilderWithKeys<Object> hashKeys() {
085    return hashKeys(DEFAULT_EXPECTED_KEYS);
086  }
087
088  /**
089   * Uses a {@link HashMap} to map keys to value collections, initialized to expect the specified
090   * number of keys.
091   *
092   * @throws IllegalArgumentException if {@code expectedKeys < 0}
093   */
094  public static MultimapBuilderWithKeys<Object> hashKeys(final int expectedKeys) {
095    checkNonnegative(expectedKeys, "expectedKeys");
096    return new MultimapBuilderWithKeys<Object>() {
097      @Override
098      <K, V> Map<K, Collection<V>> createMap() {
099        return Maps.newHashMapWithExpectedSize(expectedKeys);
100      }
101    };
102  }
103
104  /**
105   * Uses a {@link LinkedHashMap} to map keys to value collections.
106   *
107   * <p>The collections returned by {@link Multimap#keySet()}, {@link Multimap#keys()}, and
108   * {@link Multimap#asMap()} will iterate through the keys in the order that they were first added
109   * to the multimap, save that if all values associated with a key are removed and then the key is
110   * added back into the multimap, that key will come last in the key iteration order.
111   */
112  public static MultimapBuilderWithKeys<Object> linkedHashKeys() {
113    return linkedHashKeys(DEFAULT_EXPECTED_KEYS);
114  }
115
116  /**
117   * Uses a {@link LinkedHashMap} to map keys to value collections, initialized to expect the
118   * specified number of keys.
119   *
120   * <p>The collections returned by {@link Multimap#keySet()}, {@link Multimap#keys()}, and
121   * {@link Multimap#asMap()} will iterate through the keys in the order that they were first added
122   * to the multimap, save that if all values associated with a key are removed and then the key is
123   * added back into the multimap, that key will come last in the key iteration order.
124   */
125  public static MultimapBuilderWithKeys<Object> linkedHashKeys(final int expectedKeys) {
126    checkNonnegative(expectedKeys, "expectedKeys");
127    return new MultimapBuilderWithKeys<Object>() {
128      @Override
129      <K, V> Map<K, Collection<V>> createMap() {
130        return newLinkedHashMapWithExpectedSize(expectedKeys);
131      }
132    };
133  }
134
135  /**
136   * Uses a naturally-ordered {@link TreeMap} to map keys to value collections.
137   *
138   * <p>The collections returned by {@link Multimap#keySet()}, {@link Multimap#keys()}, and
139   * {@link Multimap#asMap()} will iterate through the keys in sorted order.
140   *
141   * <p>For all multimaps generated by the resulting builder, the {@link Multimap#keySet()} can be
142   * safely cast to a {@link java.util.SortedSet}, and the {@link Multimap#asMap()} can safely be
143   * cast to a {@link java.util.SortedMap}.
144   */
145  @SuppressWarnings("rawtypes")
146  public static MultimapBuilderWithKeys<Comparable> treeKeys() {
147    return treeKeys(Ordering.natural());
148  }
149
150  /**
151   * Uses a {@link TreeMap} sorted by the specified comparator to map keys to value collections.
152   *
153   * <p>The collections returned by {@link Multimap#keySet()}, {@link Multimap#keys()}, and
154   * {@link Multimap#asMap()} will iterate through the keys in sorted order.
155   *
156   * <p>For all multimaps generated by the resulting builder, the {@link Multimap#keySet()} can be
157   * safely cast to a {@link java.util.SortedSet}, and the {@link Multimap#asMap()} can safely be
158   * cast to a {@link java.util.SortedMap}.
159   *
160   * <p>Multimaps generated by the resulting builder will not be serializable if {@code comparator}
161   * is not serializable.
162   */
163  public static <K0> MultimapBuilderWithKeys<K0> treeKeys(final Comparator<K0> comparator) {
164    checkNotNull(comparator);
165    return new MultimapBuilderWithKeys<K0>() {
166      @Override
167      <K extends K0, V> Map<K, Collection<V>> createMap() {
168        return new TreeMap<>(comparator);
169      }
170    };
171  }
172
173  /**
174   * Uses an {@link EnumMap} to map keys to value collections.
175   *
176   * @since 16.0
177   */
178  public static <K0 extends Enum<K0>> MultimapBuilderWithKeys<K0> enumKeys(
179      final Class<K0> keyClass) {
180    checkNotNull(keyClass);
181    return new MultimapBuilderWithKeys<K0>() {
182      @SuppressWarnings("unchecked")
183      @Override
184      <K extends K0, V> Map<K, Collection<V>> createMap() {
185        // K must actually be K0, since enums are effectively final
186        // (their subclasses are inaccessible)
187        return (Map<K, Collection<V>>) new EnumMap<K0, Collection<V>>(keyClass);
188      }
189    };
190  }
191
192  private static final class ArrayListSupplier<V> implements Supplier<List<V>>, Serializable {
193    private final int expectedValuesPerKey;
194
195    ArrayListSupplier(int expectedValuesPerKey) {
196      this.expectedValuesPerKey = checkNonnegative(expectedValuesPerKey, "expectedValuesPerKey");
197    }
198
199    @Override
200    public List<V> get() {
201      return new ArrayList<V>(expectedValuesPerKey);
202    }
203  }
204
205  private enum LinkedListSupplier implements Supplier<List<Object>> {
206    INSTANCE;
207
208    public static <V> Supplier<List<V>> instance() {
209      // Each call generates a fresh LinkedList, which can serve as a List<V> for any V.
210      @SuppressWarnings({"rawtypes", "unchecked"})
211      Supplier<List<V>> result = (Supplier) INSTANCE;
212      return result;
213    }
214
215    @Override
216    public List<Object> get() {
217      return new LinkedList<>();
218    }
219  }
220
221  private static final class HashSetSupplier<V> implements Supplier<Set<V>>, Serializable {
222    private final int expectedValuesPerKey;
223
224    HashSetSupplier(int expectedValuesPerKey) {
225      this.expectedValuesPerKey = checkNonnegative(expectedValuesPerKey, "expectedValuesPerKey");
226    }
227
228    @Override
229    public Set<V> get() {
230      return Sets.newHashSetWithExpectedSize(expectedValuesPerKey);
231    }
232  }
233
234  private static final class LinkedHashSetSupplier<V> implements Supplier<Set<V>>, Serializable {
235    private final int expectedValuesPerKey;
236
237    LinkedHashSetSupplier(int expectedValuesPerKey) {
238      this.expectedValuesPerKey = checkNonnegative(expectedValuesPerKey, "expectedValuesPerKey");
239    }
240
241    @Override
242    public Set<V> get() {
243      return Sets.newLinkedHashSetWithExpectedSize(expectedValuesPerKey);
244    }
245  }
246
247  private static final class TreeSetSupplier<V> implements Supplier<SortedSet<V>>, Serializable {
248    private final Comparator<? super V> comparator;
249
250    TreeSetSupplier(Comparator<? super V> comparator) {
251      this.comparator = checkNotNull(comparator);
252    }
253
254    @Override
255    public SortedSet<V> get() {
256      return new TreeSet<V>(comparator);
257    }
258  }
259
260  private static final class EnumSetSupplier<V extends Enum<V>>
261      implements Supplier<Set<V>>, Serializable {
262    private final Class<V> clazz;
263
264    EnumSetSupplier(Class<V> clazz) {
265      this.clazz = checkNotNull(clazz);
266    }
267
268    @Override
269    public Set<V> get() {
270      return EnumSet.noneOf(clazz);
271    }
272  }
273
274  /**
275   * An intermediate stage in a {@link MultimapBuilder} in which the key-value collection map
276   * implementation has been specified, but the value collection implementation has not.
277   *
278   * @param <K0> The upper bound on the key type of the generated multimap.
279   *
280   * @since 16.0
281   */
282  public abstract static class MultimapBuilderWithKeys<K0> {
283
284    private static final int DEFAULT_EXPECTED_VALUES_PER_KEY = 2;
285
286    MultimapBuilderWithKeys() {}
287
288    abstract <K extends K0, V> Map<K, Collection<V>> createMap();
289
290    /**
291     * Uses an {@link ArrayList} to store value collections.
292     */
293    public ListMultimapBuilder<K0, Object> arrayListValues() {
294      return arrayListValues(DEFAULT_EXPECTED_VALUES_PER_KEY);
295    }
296
297    /**
298     * Uses an {@link ArrayList} to store value collections, initialized to expect the specified
299     * number of values per key.
300     *
301     * @throws IllegalArgumentException if {@code expectedValuesPerKey < 0}
302     */
303    public ListMultimapBuilder<K0, Object> arrayListValues(final int expectedValuesPerKey) {
304      checkNonnegative(expectedValuesPerKey, "expectedValuesPerKey");
305      return new ListMultimapBuilder<K0, Object>() {
306        @Override
307        public <K extends K0, V> ListMultimap<K, V> build() {
308          return Multimaps.newListMultimap(
309              MultimapBuilderWithKeys.this.<K, V>createMap(),
310              new ArrayListSupplier<V>(expectedValuesPerKey));
311        }
312      };
313    }
314
315    /**
316     * Uses a {@link LinkedList} to store value collections.
317     */
318    public ListMultimapBuilder<K0, Object> linkedListValues() {
319      return new ListMultimapBuilder<K0, Object>() {
320        @Override
321        public <K extends K0, V> ListMultimap<K, V> build() {
322          return Multimaps.newListMultimap(
323              MultimapBuilderWithKeys.this.<K, V>createMap(), LinkedListSupplier.<V>instance());
324        }
325      };
326    }
327
328    /**
329     * Uses a {@link HashSet} to store value collections.
330     */
331    public SetMultimapBuilder<K0, Object> hashSetValues() {
332      return hashSetValues(DEFAULT_EXPECTED_VALUES_PER_KEY);
333    }
334
335    /**
336     * Uses a {@link HashSet} to store value collections, initialized to expect the specified number
337     * of values per key.
338     *
339     * @throws IllegalArgumentException if {@code expectedValuesPerKey < 0}
340     */
341    public SetMultimapBuilder<K0, Object> hashSetValues(final int expectedValuesPerKey) {
342      checkNonnegative(expectedValuesPerKey, "expectedValuesPerKey");
343      return new SetMultimapBuilder<K0, Object>() {
344        @Override
345        public <K extends K0, V> SetMultimap<K, V> build() {
346          return Multimaps.newSetMultimap(
347              MultimapBuilderWithKeys.this.<K, V>createMap(),
348              new HashSetSupplier<V>(expectedValuesPerKey));
349        }
350      };
351    }
352
353    /**
354     * Uses a {@link LinkedHashSet} to store value collections.
355     */
356    public SetMultimapBuilder<K0, Object> linkedHashSetValues() {
357      return linkedHashSetValues(DEFAULT_EXPECTED_VALUES_PER_KEY);
358    }
359
360    /**
361     * Uses a {@link LinkedHashSet} to store value collections, initialized to expect the specified
362     * number of values per key.
363     *
364     * @throws IllegalArgumentException if {@code expectedValuesPerKey < 0}
365     */
366    public SetMultimapBuilder<K0, Object> linkedHashSetValues(final int expectedValuesPerKey) {
367      checkNonnegative(expectedValuesPerKey, "expectedValuesPerKey");
368      return new SetMultimapBuilder<K0, Object>() {
369        @Override
370        public <K extends K0, V> SetMultimap<K, V> build() {
371          return Multimaps.newSetMultimap(
372              MultimapBuilderWithKeys.this.<K, V>createMap(),
373              new LinkedHashSetSupplier<V>(expectedValuesPerKey));
374        }
375      };
376    }
377
378    /**
379     * Uses a naturally-ordered {@link TreeSet} to store value collections.
380     */
381    @SuppressWarnings("rawtypes")
382    public SortedSetMultimapBuilder<K0, Comparable> treeSetValues() {
383      return treeSetValues(Ordering.natural());
384    }
385
386    /**
387     * Uses a {@link TreeSet} ordered by the specified comparator to store value collections.
388     *
389     * <p>Multimaps generated by the resulting builder will not be serializable if
390     * {@code comparator} is not serializable.
391     */
392    public <V0> SortedSetMultimapBuilder<K0, V0> treeSetValues(final Comparator<V0> comparator) {
393      checkNotNull(comparator, "comparator");
394      return new SortedSetMultimapBuilder<K0, V0>() {
395        @Override
396        public <K extends K0, V extends V0> SortedSetMultimap<K, V> build() {
397          return Multimaps.newSortedSetMultimap(
398              MultimapBuilderWithKeys.this.<K, V>createMap(), new TreeSetSupplier<V>(comparator));
399        }
400      };
401    }
402
403    /**
404     * Uses an {@link EnumSet} to store value collections.
405     */
406    public <V0 extends Enum<V0>> SetMultimapBuilder<K0, V0> enumSetValues(
407        final Class<V0> valueClass) {
408      checkNotNull(valueClass, "valueClass");
409      return new SetMultimapBuilder<K0, V0>() {
410        @Override
411        public <K extends K0, V extends V0> SetMultimap<K, V> build() {
412          // V must actually be V0, since enums are effectively final
413          // (their subclasses are inaccessible)
414          @SuppressWarnings({"unchecked", "rawtypes"})
415          Supplier<Set<V>> factory = (Supplier) new EnumSetSupplier<V0>(valueClass);
416          return Multimaps.newSetMultimap(MultimapBuilderWithKeys.this.<K, V>createMap(), factory);
417        }
418      };
419    }
420  }
421
422  /**
423   * Returns a new, empty {@code Multimap} with the specified implementation.
424   */
425  public abstract <K extends K0, V extends V0> Multimap<K, V> build();
426
427  /**
428   * Returns a {@code Multimap} with the specified implementation, initialized with the entries of
429   * {@code multimap}.
430   */
431  public <K extends K0, V extends V0> Multimap<K, V> build(
432      Multimap<? extends K, ? extends V> multimap) {
433    Multimap<K, V> result = build();
434    result.putAll(multimap);
435    return result;
436  }
437
438  /**
439   * A specialization of {@link MultimapBuilder} that generates {@link ListMultimap} instances.
440   *
441   * @since 16.0
442   */
443  public abstract static class ListMultimapBuilder<K0, V0> extends MultimapBuilder<K0, V0> {
444    ListMultimapBuilder() {}
445
446    @Override
447    public abstract <K extends K0, V extends V0> ListMultimap<K, V> build();
448
449    @Override
450    public <K extends K0, V extends V0> ListMultimap<K, V> build(
451        Multimap<? extends K, ? extends V> multimap) {
452      return (ListMultimap<K, V>) super.build(multimap);
453    }
454  }
455
456  /**
457   * A specialization of {@link MultimapBuilder} that generates {@link SetMultimap} instances.
458   *
459   * @since 16.0
460   */
461  public abstract static class SetMultimapBuilder<K0, V0> extends MultimapBuilder<K0, V0> {
462    SetMultimapBuilder() {}
463
464    @Override
465    public abstract <K extends K0, V extends V0> SetMultimap<K, V> build();
466
467    @Override
468    public <K extends K0, V extends V0> SetMultimap<K, V> build(
469        Multimap<? extends K, ? extends V> multimap) {
470      return (SetMultimap<K, V>) super.build(multimap);
471    }
472  }
473
474  /**
475   * A specialization of {@link MultimapBuilder} that generates {@link SortedSetMultimap} instances.
476   *
477   * @since 16.0
478   */
479  public abstract static class SortedSetMultimapBuilder<K0, V0> extends SetMultimapBuilder<K0, V0> {
480    SortedSetMultimapBuilder() {}
481
482    @Override
483    public abstract <K extends K0, V extends V0> SortedSetMultimap<K, V> build();
484
485    @Override
486    public <K extends K0, V extends V0> SortedSetMultimap<K, V> build(
487        Multimap<? extends K, ? extends V> multimap) {
488      return (SortedSetMultimap<K, V>) super.build(multimap);
489    }
490  }
491}