001/* 002 * Copyright (C) 2008 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.base.Preconditions.checkState; 021import static com.google.common.collect.CollectPreconditions.checkEntryNotNull; 022 023import com.google.common.annotations.Beta; 024import com.google.common.annotations.GwtCompatible; 025import com.google.j2objc.annotations.WeakOuter; 026 027import java.io.Serializable; 028import java.util.Arrays; 029import java.util.Collections; 030import java.util.Comparator; 031import java.util.EnumMap; 032import java.util.Iterator; 033import java.util.Map; 034 035import javax.annotation.Nullable; 036 037/** 038 * A {@link Map} whose contents will never change, with many other important properties detailed at 039 * {@link ImmutableCollection}. 040 * 041 * <p>See the Guava User Guide article on <a href= 042 * "https://github.com/google/guava/wiki/ImmutableCollectionsExplained"> 043 * immutable collections</a>. 044 * 045 * @author Jesse Wilson 046 * @author Kevin Bourrillion 047 * @since 2.0 048 */ 049@GwtCompatible(serializable = true, emulated = true) 050@SuppressWarnings("serial") // we're overriding default serialization 051public abstract class ImmutableMap<K, V> implements Map<K, V>, Serializable { 052 053 /** 054 * Returns the empty map. This map behaves and performs comparably to 055 * {@link Collections#emptyMap}, and is preferable mainly for consistency 056 * and maintainability of your code. 057 */ 058 public static <K, V> ImmutableMap<K, V> of() { 059 return ImmutableBiMap.of(); 060 } 061 062 /** 063 * Returns an immutable map containing a single entry. This map behaves and 064 * performs comparably to {@link Collections#singletonMap} but will not accept 065 * a null key or value. It is preferable mainly for consistency and 066 * maintainability of your code. 067 */ 068 public static <K, V> ImmutableMap<K, V> of(K k1, V v1) { 069 return ImmutableBiMap.of(k1, v1); 070 } 071 072 /** 073 * Returns an immutable map containing the given entries, in order. 074 * 075 * @throws IllegalArgumentException if duplicate keys are provided 076 */ 077 public static <K, V> ImmutableMap<K, V> of(K k1, V v1, K k2, V v2) { 078 return RegularImmutableMap.fromEntries(entryOf(k1, v1), entryOf(k2, v2)); 079 } 080 081 /** 082 * Returns an immutable map containing the given entries, in order. 083 * 084 * @throws IllegalArgumentException if duplicate keys are provided 085 */ 086 public static <K, V> ImmutableMap<K, V> of(K k1, V v1, K k2, V v2, K k3, V v3) { 087 return RegularImmutableMap.fromEntries(entryOf(k1, v1), entryOf(k2, v2), entryOf(k3, v3)); 088 } 089 090 /** 091 * Returns an immutable map containing the given entries, in order. 092 * 093 * @throws IllegalArgumentException if duplicate keys are provided 094 */ 095 public static <K, V> ImmutableMap<K, V> of(K k1, V v1, K k2, V v2, K k3, V v3, K k4, V v4) { 096 return RegularImmutableMap.fromEntries( 097 entryOf(k1, v1), entryOf(k2, v2), entryOf(k3, v3), entryOf(k4, v4)); 098 } 099 100 /** 101 * Returns an immutable map containing the given entries, in order. 102 * 103 * @throws IllegalArgumentException if duplicate keys are provided 104 */ 105 public static <K, V> ImmutableMap<K, V> of( 106 K k1, V v1, K k2, V v2, K k3, V v3, K k4, V v4, K k5, V v5) { 107 return RegularImmutableMap.fromEntries( 108 entryOf(k1, v1), entryOf(k2, v2), entryOf(k3, v3), entryOf(k4, v4), entryOf(k5, v5)); 109 } 110 111 // looking for of() with > 5 entries? Use the builder instead. 112 113 /** 114 * Verifies that {@code key} and {@code value} are non-null, and returns a new 115 * immutable entry with those values. 116 * 117 * <p>A call to {@link Map.Entry#setValue} on the returned entry will always 118 * throw {@link UnsupportedOperationException}. 119 */ 120 static <K, V> ImmutableMapEntry<K, V> entryOf(K key, V value) { 121 return new ImmutableMapEntry<K, V>(key, value); 122 } 123 124 /** 125 * Returns a new builder. The generated builder is equivalent to the builder 126 * created by the {@link Builder} constructor. 127 */ 128 public static <K, V> Builder<K, V> builder() { 129 return new Builder<K, V>(); 130 } 131 132 static void checkNoConflict( 133 boolean safe, String conflictDescription, Entry<?, ?> entry1, Entry<?, ?> entry2) { 134 if (!safe) { 135 throw new IllegalArgumentException( 136 "Multiple entries with same " + conflictDescription + ": " + entry1 + " and " + entry2); 137 } 138 } 139 140 /** 141 * A builder for creating immutable map instances, especially {@code public 142 * static final} maps ("constant maps"). Example: <pre> {@code 143 * 144 * static final ImmutableMap<String, Integer> WORD_TO_INT = 145 * new ImmutableMap.Builder<String, Integer>() 146 * .put("one", 1) 147 * .put("two", 2) 148 * .put("three", 3) 149 * .build();}</pre> 150 * 151 * <p>For <i>small</i> immutable maps, the {@code ImmutableMap.of()} methods are 152 * even more convenient. 153 * 154 * <p>Builder instances can be reused - it is safe to call {@link #build} 155 * multiple times to build multiple maps in series. Each map is a superset of 156 * the maps created before it. 157 * 158 * @since 2.0 159 */ 160 public static class Builder<K, V> { 161 Comparator<? super V> valueComparator; 162 ImmutableMapEntry<K, V>[] entries; 163 int size; 164 boolean entriesUsed; 165 166 /** 167 * Creates a new builder. The returned builder is equivalent to the builder 168 * generated by {@link ImmutableMap#builder}. 169 */ 170 public Builder() { 171 this(ImmutableCollection.Builder.DEFAULT_INITIAL_CAPACITY); 172 } 173 174 @SuppressWarnings("unchecked") 175 Builder(int initialCapacity) { 176 this.entries = new ImmutableMapEntry[initialCapacity]; 177 this.size = 0; 178 this.entriesUsed = false; 179 } 180 181 private void ensureCapacity(int minCapacity) { 182 if (minCapacity > entries.length) { 183 entries = 184 ObjectArrays.arraysCopyOf( 185 entries, ImmutableCollection.Builder.expandedCapacity(entries.length, minCapacity)); 186 entriesUsed = false; 187 } 188 } 189 190 /** 191 * Associates {@code key} with {@code value} in the built map. Duplicate 192 * keys are not allowed, and will cause {@link #build} to fail. 193 */ 194 public Builder<K, V> put(K key, V value) { 195 ensureCapacity(size + 1); 196 ImmutableMapEntry<K, V> entry = entryOf(key, value); 197 // don't inline this: we want to fail atomically if key or value is null 198 entries[size++] = entry; 199 return this; 200 } 201 202 /** 203 * Adds the given {@code entry} to the map, making it immutable if 204 * necessary. Duplicate keys are not allowed, and will cause {@link #build} 205 * to fail. 206 * 207 * @since 11.0 208 */ 209 public Builder<K, V> put(Entry<? extends K, ? extends V> entry) { 210 return put(entry.getKey(), entry.getValue()); 211 } 212 213 /** 214 * Associates all of the given map's keys and values in the built map. 215 * Duplicate keys are not allowed, and will cause {@link #build} to fail. 216 * 217 * @throws NullPointerException if any key or value in {@code map} is null 218 */ 219 public Builder<K, V> putAll(Map<? extends K, ? extends V> map) { 220 ensureCapacity(size + map.size()); 221 return putAll(map.entrySet()); 222 } 223 224 /** 225 * Adds all of the given entries to the built map. Duplicate keys are not 226 * allowed, and will cause {@link #build} to fail. 227 * 228 * @throws NullPointerException if any key, value, or entry is null 229 * @since 19.0 230 */ 231 @Beta 232 public Builder<K, V> putAll(Iterable<? extends Entry<? extends K, ? extends V>> entries) { 233 for (Entry<? extends K, ? extends V> entry : entries) { 234 put(entry); 235 } 236 return this; 237 } 238 239 /** 240 * Configures this {@code Builder} to order entries by value according to the specified 241 * comparator. 242 * 243 * <p>The sort order is stable, that is, if two entries have values that compare 244 * as equivalent, the entry that was inserted first will be first in the built map's 245 * iteration order. 246 * 247 * @throws IllegalStateException if this method was already called 248 * @since 19.0 249 */ 250 @Beta 251 public Builder<K, V> orderEntriesByValue(Comparator<? super V> valueComparator) { 252 checkState(this.valueComparator == null, "valueComparator was already set"); 253 this.valueComparator = checkNotNull(valueComparator, "valueComparator"); 254 return this; 255 } 256 257 /* 258 * TODO(kevinb): Should build() and the ImmutableBiMap & ImmutableSortedMap 259 * versions throw an IllegalStateException instead? 260 */ 261 262 /** 263 * Returns a newly-created immutable map. 264 * 265 * @throws IllegalArgumentException if duplicate keys were added 266 */ 267 public ImmutableMap<K, V> build() { 268 switch (size) { 269 case 0: 270 return of(); 271 case 1: 272 return of(entries[0].getKey(), entries[0].getValue()); 273 default: 274 /* 275 * If entries is full, then this implementation may end up using the entries array 276 * directly and writing over the entry objects with non-terminal entries, but this is 277 * safe; if this Builder is used further, it will grow the entries array (so it can't 278 * affect the original array), and future build() calls will always copy any entry 279 * objects that cannot be safely reused. 280 */ 281 if (valueComparator != null) { 282 if (entriesUsed) { 283 entries = ObjectArrays.arraysCopyOf(entries, size); 284 } 285 Arrays.sort(entries, 0, size, 286 Ordering.from(valueComparator).onResultOf(Maps.<V>valueFunction())); 287 } 288 entriesUsed = size == entries.length; 289 return RegularImmutableMap.fromEntryArray(size, entries); 290 } 291 } 292 } 293 294 /** 295 * Returns an immutable map containing the same entries as {@code map}. If 296 * {@code map} somehow contains entries with duplicate keys (for example, if 297 * it is a {@code SortedMap} whose comparator is not <i>consistent with 298 * equals</i>), the results of this method are undefined. 299 * 300 * <p>Despite the method name, this method attempts to avoid actually copying 301 * the data when it is safe to do so. The exact circumstances under which a 302 * copy will or will not be performed are undocumented and subject to change. 303 * 304 * @throws NullPointerException if any key or value in {@code map} is null 305 */ 306 public static <K, V> ImmutableMap<K, V> copyOf(Map<? extends K, ? extends V> map) { 307 if ((map instanceof ImmutableMap) && !(map instanceof ImmutableSortedMap)) { 308 // TODO(lowasser): Make ImmutableMap.copyOf(immutableBiMap) call copyOf() 309 // on the ImmutableMap delegate(), rather than the bimap itself 310 311 @SuppressWarnings("unchecked") // safe since map is not writable 312 ImmutableMap<K, V> kvMap = (ImmutableMap<K, V>) map; 313 if (!kvMap.isPartialView()) { 314 return kvMap; 315 } 316 } else if (map instanceof EnumMap) { 317 @SuppressWarnings("unchecked") // safe since map is not writable 318 ImmutableMap<K, V> kvMap = (ImmutableMap<K, V>) copyOfEnumMap((EnumMap<?, ?>) map); 319 return kvMap; 320 } 321 return copyOf(map.entrySet()); 322 } 323 324 /** 325 * Returns an immutable map containing the specified entries. The returned 326 * map iterates over entries in the same order as the original iterable. 327 * 328 * @throws NullPointerException if any key, value, or entry is null 329 * @throws IllegalArgumentException if two entries have the same key 330 * @since 19.0 331 */ 332 @Beta 333 public static <K, V> ImmutableMap<K, V> copyOf( 334 Iterable<? extends Entry<? extends K, ? extends V>> entries) { 335 @SuppressWarnings("unchecked") // we'll only be using getKey and getValue, which are covariant 336 Entry<K, V>[] entryArray = (Entry<K, V>[]) Iterables.toArray(entries, EMPTY_ENTRY_ARRAY); 337 switch (entryArray.length) { 338 case 0: 339 return of(); 340 case 1: 341 Entry<K, V> onlyEntry = entryArray[0]; 342 return of(onlyEntry.getKey(), onlyEntry.getValue()); 343 default: 344 /* 345 * The current implementation will end up using entryArray directly, though it will write 346 * over the (arbitrary, potentially mutable) Entry objects actually stored in entryArray. 347 */ 348 return RegularImmutableMap.fromEntries(entryArray); 349 } 350 } 351 352 private static <K extends Enum<K>, V> ImmutableMap<K, V> copyOfEnumMap( 353 EnumMap<K, ? extends V> original) { 354 EnumMap<K, V> copy = new EnumMap<K, V>(original); 355 for (Map.Entry<?, ?> entry : copy.entrySet()) { 356 checkEntryNotNull(entry.getKey(), entry.getValue()); 357 } 358 return ImmutableEnumMap.asImmutable(copy); 359 } 360 361 static final Entry<?, ?>[] EMPTY_ENTRY_ARRAY = new Entry<?, ?>[0]; 362 363 abstract static class IteratorBasedImmutableMap<K, V> extends ImmutableMap<K, V> { 364 abstract UnmodifiableIterator<Entry<K, V>> entryIterator(); 365 366 @Override 367 ImmutableSet<Entry<K, V>> createEntrySet() { 368 @WeakOuter 369 class EntrySetImpl extends ImmutableMapEntrySet<K, V> { 370 @Override 371 ImmutableMap<K, V> map() { 372 return IteratorBasedImmutableMap.this; 373 } 374 375 @Override 376 public UnmodifiableIterator<Entry<K, V>> iterator() { 377 return entryIterator(); 378 } 379 } 380 return new EntrySetImpl(); 381 } 382 } 383 384 ImmutableMap() {} 385 386 /** 387 * Guaranteed to throw an exception and leave the map unmodified. 388 * 389 * @throws UnsupportedOperationException always 390 * @deprecated Unsupported operation. 391 */ 392 @Deprecated 393 @Override 394 public final V put(K k, V v) { 395 throw new UnsupportedOperationException(); 396 } 397 398 /** 399 * Guaranteed to throw an exception and leave the map unmodified. 400 * 401 * @throws UnsupportedOperationException always 402 * @deprecated Unsupported operation. 403 */ 404 @Deprecated 405 @Override 406 public final V remove(Object o) { 407 throw new UnsupportedOperationException(); 408 } 409 410 /** 411 * Guaranteed to throw an exception and leave the map unmodified. 412 * 413 * @throws UnsupportedOperationException always 414 * @deprecated Unsupported operation. 415 */ 416 @Deprecated 417 @Override 418 public final void putAll(Map<? extends K, ? extends V> map) { 419 throw new UnsupportedOperationException(); 420 } 421 422 /** 423 * Guaranteed to throw an exception and leave the map unmodified. 424 * 425 * @throws UnsupportedOperationException always 426 * @deprecated Unsupported operation. 427 */ 428 @Deprecated 429 @Override 430 public final void clear() { 431 throw new UnsupportedOperationException(); 432 } 433 434 @Override 435 public boolean isEmpty() { 436 return size() == 0; 437 } 438 439 @Override 440 public boolean containsKey(@Nullable Object key) { 441 return get(key) != null; 442 } 443 444 @Override 445 public boolean containsValue(@Nullable Object value) { 446 return values().contains(value); 447 } 448 449 // Overriding to mark it Nullable 450 @Override 451 public abstract V get(@Nullable Object key); 452 453 private transient ImmutableSet<Entry<K, V>> entrySet; 454 455 /** 456 * Returns an immutable set of the mappings in this map. The entries are in 457 * the same order as the parameters used to build this map. 458 */ 459 @Override 460 public ImmutableSet<Entry<K, V>> entrySet() { 461 ImmutableSet<Entry<K, V>> result = entrySet; 462 return (result == null) ? entrySet = createEntrySet() : result; 463 } 464 465 abstract ImmutableSet<Entry<K, V>> createEntrySet(); 466 467 private transient ImmutableSet<K> keySet; 468 469 /** 470 * Returns an immutable set of the keys in this map. These keys are in 471 * the same order as the parameters used to build this map. 472 */ 473 @Override 474 public ImmutableSet<K> keySet() { 475 ImmutableSet<K> result = keySet; 476 return (result == null) ? keySet = createKeySet() : result; 477 } 478 479 ImmutableSet<K> createKeySet() { 480 return isEmpty() ? ImmutableSet.<K>of() : new ImmutableMapKeySet<K, V>(this); 481 } 482 483 UnmodifiableIterator<K> keyIterator() { 484 final UnmodifiableIterator<Entry<K, V>> entryIterator = entrySet().iterator(); 485 return new UnmodifiableIterator<K>() { 486 @Override 487 public boolean hasNext() { 488 return entryIterator.hasNext(); 489 } 490 491 @Override 492 public K next() { 493 return entryIterator.next().getKey(); 494 } 495 }; 496 } 497 498 private transient ImmutableCollection<V> values; 499 500 /** 501 * Returns an immutable collection of the values in this map. The values are 502 * in the same order as the parameters used to build this map. 503 */ 504 @Override 505 public ImmutableCollection<V> values() { 506 ImmutableCollection<V> result = values; 507 return (result == null) ? values = new ImmutableMapValues<K, V>(this) : result; 508 } 509 510 // cached so that this.multimapView().inverse() only computes inverse once 511 private transient ImmutableSetMultimap<K, V> multimapView; 512 513 /** 514 * Returns a multimap view of the map. 515 * 516 * @since 14.0 517 */ 518 @Beta 519 public ImmutableSetMultimap<K, V> asMultimap() { 520 if (isEmpty()) { 521 return ImmutableSetMultimap.of(); 522 } 523 ImmutableSetMultimap<K, V> result = multimapView; 524 return (result == null) 525 ? (multimapView = 526 new ImmutableSetMultimap<K, V>(new MapViewOfValuesAsSingletonSets(), size(), null)) 527 : result; 528 } 529 530 @WeakOuter 531 private final class MapViewOfValuesAsSingletonSets 532 extends IteratorBasedImmutableMap<K, ImmutableSet<V>> { 533 534 @Override 535 public int size() { 536 return ImmutableMap.this.size(); 537 } 538 539 @Override 540 public ImmutableSet<K> keySet() { 541 return ImmutableMap.this.keySet(); 542 } 543 544 @Override 545 public boolean containsKey(@Nullable Object key) { 546 return ImmutableMap.this.containsKey(key); 547 } 548 549 @Override 550 public ImmutableSet<V> get(@Nullable Object key) { 551 V outerValue = ImmutableMap.this.get(key); 552 return (outerValue == null) ? null : ImmutableSet.of(outerValue); 553 } 554 555 @Override 556 boolean isPartialView() { 557 return ImmutableMap.this.isPartialView(); 558 } 559 560 @Override 561 public int hashCode() { 562 // ImmutableSet.of(value).hashCode() == value.hashCode(), so the hashes are the same 563 return ImmutableMap.this.hashCode(); 564 } 565 566 @Override 567 boolean isHashCodeFast() { 568 return ImmutableMap.this.isHashCodeFast(); 569 } 570 571 @Override 572 UnmodifiableIterator<Entry<K, ImmutableSet<V>>> entryIterator() { 573 final Iterator<Entry<K, V>> backingIterator = ImmutableMap.this.entrySet().iterator(); 574 return new UnmodifiableIterator<Entry<K, ImmutableSet<V>>>() { 575 @Override 576 public boolean hasNext() { 577 return backingIterator.hasNext(); 578 } 579 580 @Override 581 public Entry<K, ImmutableSet<V>> next() { 582 final Entry<K, V> backingEntry = backingIterator.next(); 583 return new AbstractMapEntry<K, ImmutableSet<V>>() { 584 @Override 585 public K getKey() { 586 return backingEntry.getKey(); 587 } 588 589 @Override 590 public ImmutableSet<V> getValue() { 591 return ImmutableSet.of(backingEntry.getValue()); 592 } 593 }; 594 } 595 }; 596 } 597 } 598 599 @Override 600 public boolean equals(@Nullable Object object) { 601 return Maps.equalsImpl(this, object); 602 } 603 604 abstract boolean isPartialView(); 605 606 @Override 607 public int hashCode() { 608 return Sets.hashCodeImpl(entrySet()); 609 } 610 611 boolean isHashCodeFast() { 612 return false; 613 } 614 615 @Override 616 public String toString() { 617 return Maps.toStringImpl(this); 618 } 619 620 /** 621 * Serialized type for all ImmutableMap instances. It captures the logical 622 * contents and they are reconstructed using public factory methods. This 623 * ensures that the implementation types remain as implementation details. 624 */ 625 static class SerializedForm implements Serializable { 626 private final Object[] keys; 627 private final Object[] values; 628 629 SerializedForm(ImmutableMap<?, ?> map) { 630 keys = new Object[map.size()]; 631 values = new Object[map.size()]; 632 int i = 0; 633 for (Entry<?, ?> entry : map.entrySet()) { 634 keys[i] = entry.getKey(); 635 values[i] = entry.getValue(); 636 i++; 637 } 638 } 639 640 Object readResolve() { 641 Builder<Object, Object> builder = new Builder<Object, Object>(keys.length); 642 return createMap(builder); 643 } 644 645 Object createMap(Builder<Object, Object> builder) { 646 for (int i = 0; i < keys.length; i++) { 647 builder.put(keys[i], values[i]); 648 } 649 return builder.build(); 650 } 651 652 private static final long serialVersionUID = 0; 653 } 654 655 Object writeReplace() { 656 return new SerializedForm(this); 657 } 658}