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}