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.collect.CollectPreconditions.checkEntryNotNull; 021import static com.google.common.collect.CollectPreconditions.checkNonnegative; 022import static com.google.common.collect.Maps.immutableEntry; 023import static java.util.Objects.requireNonNull; 024 025import com.google.common.annotations.GwtCompatible; 026import com.google.common.annotations.GwtIncompatible; 027import com.google.common.annotations.J2ktIncompatible; 028import com.google.errorprone.annotations.CanIgnoreReturnValue; 029import com.google.errorprone.annotations.DoNotCall; 030import com.google.errorprone.annotations.DoNotMock; 031import com.google.j2objc.annotations.Weak; 032import com.google.j2objc.annotations.WeakOuter; 033import java.io.InvalidObjectException; 034import java.io.ObjectInputStream; 035import java.io.Serializable; 036import java.util.Arrays; 037import java.util.Collection; 038import java.util.Comparator; 039import java.util.Iterator; 040import java.util.Map; 041import java.util.Map.Entry; 042import java.util.Set; 043import javax.annotation.CheckForNull; 044import org.checkerframework.checker.nullness.qual.Nullable; 045 046/** 047 * A {@link Multimap} whose contents will never change, with many other important properties 048 * detailed at {@link ImmutableCollection}. 049 * 050 * <p><b>Warning:</b> avoid <i>direct</i> usage of {@link ImmutableMultimap} as a type (as with 051 * {@link Multimap} itself). Prefer subtypes such as {@link ImmutableSetMultimap} or {@link 052 * ImmutableListMultimap}, which have well-defined {@link #equals} semantics, thus avoiding a common 053 * source of bugs and confusion. 054 * 055 * <p><b>Note:</b> every {@link ImmutableMultimap} offers an {@link #inverse} view, so there is no 056 * need for a distinct {@code ImmutableBiMultimap} type. 057 * 058 * <p><a id="iteration"></a> 059 * 060 * <p><b>Key-grouped iteration.</b> All view collections follow the same iteration order. In all 061 * current implementations, the iteration order always keeps multiple entries with the same key 062 * together. Any creation method that would customarily respect insertion order (such as {@link 063 * #copyOf(Multimap)}) instead preserves key-grouped order by inserting entries for an existing key 064 * immediately after the last entry having that key. 065 * 066 * <p>See the Guava User Guide article on <a href= 067 * "https://github.com/google/guava/wiki/ImmutableCollectionsExplained">immutable collections</a>. 068 * 069 * @author Jared Levy 070 * @since 2.0 071 */ 072@GwtCompatible(emulated = true) 073@ElementTypesAreNonnullByDefault 074public abstract class ImmutableMultimap<K, V> extends BaseImmutableMultimap<K, V> 075 implements Serializable { 076 077 /** 078 * Returns an empty multimap. 079 * 080 * <p><b>Performance note:</b> the instance returned is a singleton. 081 */ 082 public static <K, V> ImmutableMultimap<K, V> of() { 083 return ImmutableListMultimap.of(); 084 } 085 086 /** Returns an immutable multimap containing a single entry. */ 087 public static <K, V> ImmutableMultimap<K, V> of(K k1, V v1) { 088 return ImmutableListMultimap.of(k1, v1); 089 } 090 091 /** Returns an immutable multimap containing the given entries, in order. */ 092 public static <K, V> ImmutableMultimap<K, V> of(K k1, V v1, K k2, V v2) { 093 return ImmutableListMultimap.of(k1, v1, k2, v2); 094 } 095 096 /** 097 * Returns an immutable multimap containing the given entries, in the "key-grouped" insertion 098 * order described in the <a href="#iteration">class documentation</a>. 099 */ 100 public static <K, V> ImmutableMultimap<K, V> of(K k1, V v1, K k2, V v2, K k3, V v3) { 101 return ImmutableListMultimap.of(k1, v1, k2, v2, k3, v3); 102 } 103 104 /** 105 * Returns an immutable multimap containing the given entries, in the "key-grouped" insertion 106 * order described in the <a href="#iteration">class documentation</a>. 107 */ 108 public static <K, V> ImmutableMultimap<K, V> of(K k1, V v1, K k2, V v2, K k3, V v3, K k4, V v4) { 109 return ImmutableListMultimap.of(k1, v1, k2, v2, k3, v3, k4, v4); 110 } 111 112 /** 113 * Returns an immutable multimap containing the given entries, in the "key-grouped" insertion 114 * order described in the <a href="#iteration">class documentation</a>. 115 */ 116 public static <K, V> ImmutableMultimap<K, V> of( 117 K k1, V v1, K k2, V v2, K k3, V v3, K k4, V v4, K k5, V v5) { 118 return ImmutableListMultimap.of(k1, v1, k2, v2, k3, v3, k4, v4, k5, v5); 119 } 120 121 // looking for of() with > 5 entries? Use the builder instead. 122 123 /** 124 * Returns a new builder. The generated builder is equivalent to the builder created by the {@link 125 * Builder} constructor. 126 */ 127 public static <K, V> Builder<K, V> builder() { 128 return new Builder<>(); 129 } 130 131 /** 132 * Returns a new builder with a hint for how many distinct keys are expected to be added. The 133 * generated builder is equivalent to that returned by {@link #builder}, but may perform better if 134 * {@code expectedKeys} is a good estimate. 135 * 136 * @throws IllegalArgumentException if {@code expectedKeys} is negative 137 * @since 33.3.0 138 */ 139 public static <K, V> Builder<K, V> builderWithExpectedKeys(int expectedKeys) { 140 checkNonnegative(expectedKeys, "expectedKeys"); 141 return new Builder<>(expectedKeys); 142 } 143 144 /** 145 * A builder for creating immutable multimap instances, especially {@code public static final} 146 * multimaps ("constant multimaps"). Example: 147 * 148 * <pre>{@code 149 * static final Multimap<String, Integer> STRING_TO_INTEGER_MULTIMAP = 150 * new ImmutableMultimap.Builder<String, Integer>() 151 * .put("one", 1) 152 * .putAll("several", 1, 2, 3) 153 * .putAll("many", 1, 2, 3, 4, 5) 154 * .build(); 155 * }</pre> 156 * 157 * <p>Builder instances can be reused; it is safe to call {@link #build} multiple times to build 158 * multiple multimaps in series. Each multimap contains the key-value mappings in the previously 159 * created multimaps. 160 * 161 * @since 2.0 162 */ 163 @DoNotMock 164 public static class Builder<K, V> { 165 @CheckForNull Map<K, ImmutableCollection.Builder<V>> builderMap; 166 @CheckForNull Comparator<? super K> keyComparator; 167 @CheckForNull Comparator<? super V> valueComparator; 168 int expectedValuesPerKey = ImmutableCollection.Builder.DEFAULT_INITIAL_CAPACITY; 169 170 /** 171 * Creates a new builder. The returned builder is equivalent to the builder generated by {@link 172 * ImmutableMultimap#builder}. 173 */ 174 public Builder() {} 175 176 /** Creates a new builder with a hint for the number of distinct keys. */ 177 Builder(int expectedKeys) { 178 if (expectedKeys > 0) { 179 builderMap = Platform.preservesInsertionOrderOnPutsMapWithExpectedSize(expectedKeys); 180 } 181 // otherwise, leave it null to be constructed lazily 182 } 183 184 Map<K, ImmutableCollection.Builder<V>> ensureBuilderMapNonNull() { 185 Map<K, ImmutableCollection.Builder<V>> result = builderMap; 186 if (result == null) { 187 result = Platform.preservesInsertionOrderOnPutsMap(); 188 builderMap = result; 189 } 190 return result; 191 } 192 193 ImmutableCollection.Builder<V> newValueCollectionBuilderWithExpectedSize(int expectedSize) { 194 return ImmutableList.builderWithExpectedSize(expectedSize); 195 } 196 197 /** 198 * Provides a hint for how many values will be associated with each key newly added to the 199 * builder after this call. This does not change semantics, but may improve performance if 200 * {@code expectedValuesPerKey} is a good estimate. 201 * 202 * <p>This may be called more than once; each newly added key will use the most recent call to 203 * {@link #expectedValuesPerKey} as its hint. 204 * 205 * @throws IllegalArgumentException if {@code expectedValuesPerKey} is negative 206 * @since 33.3.0 207 */ 208 @CanIgnoreReturnValue 209 public Builder<K, V> expectedValuesPerKey(int expectedValuesPerKey) { 210 checkNonnegative(expectedValuesPerKey, "expectedValuesPerKey"); 211 212 // Always presize to at least 1, since we only bother creating a value collection if there's 213 // at least one element. 214 this.expectedValuesPerKey = Math.max(expectedValuesPerKey, 1); 215 216 return this; 217 } 218 219 /** 220 * By default, if we are handed a value collection bigger than expectedValuesPerKey, presize to 221 * accept that many elements. 222 * 223 * <p>This gets overridden in ImmutableSetMultimap.Builder to only trust the size of {@code 224 * values} if it is a Set and therefore probably already deduplicated. 225 */ 226 int expectedValueCollectionSize(int defaultExpectedValues, Iterable<?> values) { 227 if (values instanceof Collection<?>) { 228 Collection<?> collection = (Collection<?>) values; 229 return Math.max(defaultExpectedValues, collection.size()); 230 } else { 231 return defaultExpectedValues; 232 } 233 } 234 235 /** Adds a key-value mapping to the built multimap. */ 236 @CanIgnoreReturnValue 237 public Builder<K, V> put(K key, V value) { 238 checkEntryNotNull(key, value); 239 ImmutableCollection.Builder<V> valuesBuilder = ensureBuilderMapNonNull().get(key); 240 if (valuesBuilder == null) { 241 valuesBuilder = newValueCollectionBuilderWithExpectedSize(expectedValuesPerKey); 242 ensureBuilderMapNonNull().put(key, valuesBuilder); 243 } 244 valuesBuilder.add(value); 245 return this; 246 } 247 248 /** 249 * Adds an entry to the built multimap. 250 * 251 * @since 11.0 252 */ 253 @CanIgnoreReturnValue 254 public Builder<K, V> put(Entry<? extends K, ? extends V> entry) { 255 return put(entry.getKey(), entry.getValue()); 256 } 257 258 /** 259 * Adds entries to the built multimap. 260 * 261 * @since 19.0 262 */ 263 @CanIgnoreReturnValue 264 public Builder<K, V> putAll(Iterable<? extends Entry<? extends K, ? extends V>> entries) { 265 for (Entry<? extends K, ? extends V> entry : entries) { 266 put(entry); 267 } 268 return this; 269 } 270 271 /** 272 * Stores a collection of values with the same key in the built multimap. 273 * 274 * @throws NullPointerException if {@code key}, {@code values}, or any element in {@code values} 275 * is null. The builder is left in an invalid state. 276 */ 277 @CanIgnoreReturnValue 278 public Builder<K, V> putAll(K key, Iterable<? extends V> values) { 279 if (key == null) { 280 throw new NullPointerException("null key in entry: null=" + Iterables.toString(values)); 281 } 282 Iterator<? extends V> valuesItr = values.iterator(); 283 if (!valuesItr.hasNext()) { 284 return this; 285 } 286 ImmutableCollection.Builder<V> valuesBuilder = ensureBuilderMapNonNull().get(key); 287 if (valuesBuilder == null) { 288 valuesBuilder = 289 newValueCollectionBuilderWithExpectedSize( 290 expectedValueCollectionSize(expectedValuesPerKey, values)); 291 ensureBuilderMapNonNull().put(key, valuesBuilder); 292 } 293 while (valuesItr.hasNext()) { 294 V value = valuesItr.next(); 295 checkEntryNotNull(key, value); 296 valuesBuilder.add(value); 297 } 298 return this; 299 } 300 301 /** 302 * Stores an array of values with the same key in the built multimap. 303 * 304 * @throws NullPointerException if the key or any value is null. The builder is left in an 305 * invalid state. 306 */ 307 @CanIgnoreReturnValue 308 public Builder<K, V> putAll(K key, V... values) { 309 return putAll(key, Arrays.asList(values)); 310 } 311 312 /** 313 * Stores another multimap's entries in the built multimap. The generated multimap's key and 314 * value orderings correspond to the iteration ordering of the {@code multimap.asMap()} view, 315 * with new keys and values following any existing keys and values. 316 * 317 * @throws NullPointerException if any key or value in {@code multimap} is null. The builder is 318 * left in an invalid state. 319 */ 320 @CanIgnoreReturnValue 321 public Builder<K, V> putAll(Multimap<? extends K, ? extends V> multimap) { 322 for (Entry<? extends K, ? extends Collection<? extends V>> entry : 323 multimap.asMap().entrySet()) { 324 putAll(entry.getKey(), entry.getValue()); 325 } 326 return this; 327 } 328 329 /** 330 * Specifies the ordering of the generated multimap's keys. 331 * 332 * @since 8.0 333 */ 334 @CanIgnoreReturnValue 335 public Builder<K, V> orderKeysBy(Comparator<? super K> keyComparator) { 336 this.keyComparator = checkNotNull(keyComparator); 337 return this; 338 } 339 340 /** 341 * Specifies the ordering of the generated multimap's values for each key. 342 * 343 * @since 8.0 344 */ 345 @CanIgnoreReturnValue 346 public Builder<K, V> orderValuesBy(Comparator<? super V> valueComparator) { 347 this.valueComparator = checkNotNull(valueComparator); 348 return this; 349 } 350 351 @CanIgnoreReturnValue 352 Builder<K, V> combine(Builder<K, V> other) { 353 if (other.builderMap != null) { 354 for (Map.Entry<K, ImmutableCollection.Builder<V>> entry : other.builderMap.entrySet()) { 355 putAll(entry.getKey(), entry.getValue().build()); 356 } 357 } 358 return this; 359 } 360 361 /** Returns a newly-created immutable multimap. */ 362 public ImmutableMultimap<K, V> build() { 363 if (builderMap == null) { 364 return ImmutableListMultimap.of(); 365 } 366 Collection<Map.Entry<K, ImmutableCollection.Builder<V>>> mapEntries = builderMap.entrySet(); 367 if (keyComparator != null) { 368 mapEntries = Ordering.from(keyComparator).<K>onKeys().immutableSortedCopy(mapEntries); 369 } 370 return ImmutableListMultimap.fromMapBuilderEntries(mapEntries, valueComparator); 371 } 372 } 373 374 /** 375 * Returns an immutable multimap containing the same mappings as {@code multimap}, in the 376 * "key-grouped" iteration order described in the class documentation. 377 * 378 * <p>Despite the method name, this method attempts to avoid actually copying the data when it is 379 * safe to do so. The exact circumstances under which a copy will or will not be performed are 380 * undocumented and subject to change. 381 * 382 * @throws NullPointerException if any key or value in {@code multimap} is null 383 */ 384 public static <K, V> ImmutableMultimap<K, V> copyOf(Multimap<? extends K, ? extends V> multimap) { 385 if (multimap instanceof ImmutableMultimap) { 386 @SuppressWarnings("unchecked") // safe since multimap is not writable 387 ImmutableMultimap<K, V> kvMultimap = (ImmutableMultimap<K, V>) multimap; 388 if (!kvMultimap.isPartialView()) { 389 return kvMultimap; 390 } 391 } 392 return ImmutableListMultimap.copyOf(multimap); 393 } 394 395 /** 396 * Returns an immutable multimap containing the specified entries. The returned multimap iterates 397 * over keys in the order they were first encountered in the input, and the values for each key 398 * are iterated in the order they were encountered. 399 * 400 * @throws NullPointerException if any key, value, or entry is null 401 * @since 19.0 402 */ 403 public static <K, V> ImmutableMultimap<K, V> copyOf( 404 Iterable<? extends Entry<? extends K, ? extends V>> entries) { 405 return ImmutableListMultimap.copyOf(entries); 406 } 407 408 final transient ImmutableMap<K, ? extends ImmutableCollection<V>> map; 409 final transient int size; 410 411 // These constants allow the deserialization code to set final fields. This 412 // holder class makes sure they are not initialized unless an instance is 413 // deserialized. 414 @GwtIncompatible // java serialization is not supported 415 @J2ktIncompatible 416 static class FieldSettersHolder { 417 static final Serialization.FieldSetter<? super ImmutableMultimap<?, ?>> MAP_FIELD_SETTER = 418 Serialization.getFieldSetter(ImmutableMultimap.class, "map"); 419 static final Serialization.FieldSetter<? super ImmutableMultimap<?, ?>> SIZE_FIELD_SETTER = 420 Serialization.getFieldSetter(ImmutableMultimap.class, "size"); 421 } 422 423 ImmutableMultimap(ImmutableMap<K, ? extends ImmutableCollection<V>> map, int size) { 424 this.map = map; 425 this.size = size; 426 } 427 428 // mutators (not supported) 429 430 /** 431 * Guaranteed to throw an exception and leave the multimap unmodified. 432 * 433 * @throws UnsupportedOperationException always 434 * @deprecated Unsupported operation. 435 */ 436 @CanIgnoreReturnValue 437 @Deprecated 438 @Override 439 @DoNotCall("Always throws UnsupportedOperationException") 440 // DoNotCall wants this to be final, but we want to override it to return more specific types. 441 // Inheritance is closed, and all subtypes are @DoNotCall, so this is safe to suppress. 442 @SuppressWarnings("DoNotCall") 443 public ImmutableCollection<V> removeAll(@CheckForNull Object key) { 444 throw new UnsupportedOperationException(); 445 } 446 447 /** 448 * Guaranteed to throw an exception and leave the multimap unmodified. 449 * 450 * @throws UnsupportedOperationException always 451 * @deprecated Unsupported operation. 452 */ 453 @CanIgnoreReturnValue 454 @Deprecated 455 @Override 456 @DoNotCall("Always throws UnsupportedOperationException") 457 // DoNotCall wants this to be final, but we want to override it to return more specific types. 458 // Inheritance is closed, and all subtypes are @DoNotCall, so this is safe to suppress. 459 @SuppressWarnings("DoNotCall") 460 public ImmutableCollection<V> replaceValues(K key, Iterable<? extends V> values) { 461 throw new UnsupportedOperationException(); 462 } 463 464 /** 465 * Guaranteed to throw an exception and leave the multimap unmodified. 466 * 467 * @throws UnsupportedOperationException always 468 * @deprecated Unsupported operation. 469 */ 470 @Deprecated 471 @Override 472 @DoNotCall("Always throws UnsupportedOperationException") 473 public final void clear() { 474 throw new UnsupportedOperationException(); 475 } 476 477 /** 478 * Returns an immutable collection of the values for the given key. If no mappings in the multimap 479 * have the provided key, an empty immutable collection is returned. The values are in the same 480 * order as the parameters used to build this multimap. 481 */ 482 @Override 483 public abstract ImmutableCollection<V> get(K key); 484 485 /** 486 * Returns an immutable multimap which is the inverse of this one. For every key-value mapping in 487 * the original, the result will have a mapping with key and value reversed. 488 * 489 * @since 11.0 490 */ 491 public abstract ImmutableMultimap<V, K> inverse(); 492 493 /** 494 * Guaranteed to throw an exception and leave the multimap unmodified. 495 * 496 * @throws UnsupportedOperationException always 497 * @deprecated Unsupported operation. 498 */ 499 @CanIgnoreReturnValue 500 @Deprecated 501 @Override 502 @DoNotCall("Always throws UnsupportedOperationException") 503 public final boolean put(K key, V value) { 504 throw new UnsupportedOperationException(); 505 } 506 507 /** 508 * Guaranteed to throw an exception and leave the multimap unmodified. 509 * 510 * @throws UnsupportedOperationException always 511 * @deprecated Unsupported operation. 512 */ 513 @CanIgnoreReturnValue 514 @Deprecated 515 @Override 516 @DoNotCall("Always throws UnsupportedOperationException") 517 public final boolean putAll(K key, Iterable<? extends V> values) { 518 throw new UnsupportedOperationException(); 519 } 520 521 /** 522 * Guaranteed to throw an exception and leave the multimap unmodified. 523 * 524 * @throws UnsupportedOperationException always 525 * @deprecated Unsupported operation. 526 */ 527 @CanIgnoreReturnValue 528 @Deprecated 529 @Override 530 @DoNotCall("Always throws UnsupportedOperationException") 531 public final boolean putAll(Multimap<? extends K, ? extends V> multimap) { 532 throw new UnsupportedOperationException(); 533 } 534 535 /** 536 * Guaranteed to throw an exception and leave the multimap unmodified. 537 * 538 * @throws UnsupportedOperationException always 539 * @deprecated Unsupported operation. 540 */ 541 @CanIgnoreReturnValue 542 @Deprecated 543 @Override 544 @DoNotCall("Always throws UnsupportedOperationException") 545 public final boolean remove(@CheckForNull Object key, @CheckForNull Object value) { 546 throw new UnsupportedOperationException(); 547 } 548 549 /** 550 * Returns {@code true} if this immutable multimap's implementation contains references to 551 * user-created objects that aren't accessible via this multimap's methods. This is generally used 552 * to determine whether {@code copyOf} implementations should make an explicit copy to avoid 553 * memory leaks. 554 */ 555 boolean isPartialView() { 556 return map.isPartialView(); 557 } 558 559 // accessors 560 561 @Override 562 public boolean containsKey(@CheckForNull Object key) { 563 return map.containsKey(key); 564 } 565 566 @Override 567 public boolean containsValue(@CheckForNull Object value) { 568 return value != null && super.containsValue(value); 569 } 570 571 @Override 572 public int size() { 573 return size; 574 } 575 576 // views 577 578 /** 579 * Returns an immutable set of the distinct keys in this multimap, in the same order as they 580 * appear in this multimap. 581 */ 582 @Override 583 public ImmutableSet<K> keySet() { 584 return map.keySet(); 585 } 586 587 @Override 588 Set<K> createKeySet() { 589 throw new AssertionError("unreachable"); 590 } 591 592 /** 593 * Returns an immutable map that associates each key with its corresponding values in the 594 * multimap. Keys and values appear in the same order as in this multimap. 595 */ 596 @Override 597 @SuppressWarnings("unchecked") // a widening cast 598 public ImmutableMap<K, Collection<V>> asMap() { 599 return (ImmutableMap) map; 600 } 601 602 @Override 603 Map<K, Collection<V>> createAsMap() { 604 throw new AssertionError("should never be called"); 605 } 606 607 /** Returns an immutable collection of all key-value pairs in the multimap. */ 608 @Override 609 public ImmutableCollection<Entry<K, V>> entries() { 610 return (ImmutableCollection<Entry<K, V>>) super.entries(); 611 } 612 613 @Override 614 ImmutableCollection<Entry<K, V>> createEntries() { 615 return new EntryCollection<>(this); 616 } 617 618 private static class EntryCollection<K, V> extends ImmutableCollection<Entry<K, V>> { 619 @Weak final ImmutableMultimap<K, V> multimap; 620 621 EntryCollection(ImmutableMultimap<K, V> multimap) { 622 this.multimap = multimap; 623 } 624 625 @Override 626 public UnmodifiableIterator<Entry<K, V>> iterator() { 627 return multimap.entryIterator(); 628 } 629 630 @Override 631 boolean isPartialView() { 632 return multimap.isPartialView(); 633 } 634 635 @Override 636 public int size() { 637 return multimap.size(); 638 } 639 640 @Override 641 public boolean contains(@CheckForNull Object object) { 642 if (object instanceof Entry) { 643 Entry<?, ?> entry = (Entry<?, ?>) object; 644 return multimap.containsEntry(entry.getKey(), entry.getValue()); 645 } 646 return false; 647 } 648 649 // redeclare to help optimizers with b/310253115 650 @SuppressWarnings("RedundantOverride") 651 @Override 652 @J2ktIncompatible // serialization 653 @GwtIncompatible // serialization 654 Object writeReplace() { 655 return super.writeReplace(); 656 } 657 658 private static final long serialVersionUID = 0; 659 } 660 661 @Override 662 UnmodifiableIterator<Entry<K, V>> entryIterator() { 663 return new UnmodifiableIterator<Entry<K, V>>() { 664 final Iterator<? extends Entry<K, ? extends ImmutableCollection<V>>> asMapItr = 665 map.entrySet().iterator(); 666 @CheckForNull K currentKey = null; 667 Iterator<V> valueItr = Iterators.emptyIterator(); 668 669 @Override 670 public boolean hasNext() { 671 return valueItr.hasNext() || asMapItr.hasNext(); 672 } 673 674 @Override 675 public Entry<K, V> next() { 676 if (!valueItr.hasNext()) { 677 Entry<K, ? extends ImmutableCollection<V>> entry = asMapItr.next(); 678 currentKey = entry.getKey(); 679 valueItr = entry.getValue().iterator(); 680 } 681 /* 682 * requireNonNull is safe: The first call to this method always enters the !hasNext() case 683 * and populates currentKey, after which it's never cleared. 684 */ 685 return immutableEntry(requireNonNull(currentKey), valueItr.next()); 686 } 687 }; 688 } 689 690 /** 691 * Returns an immutable multiset containing all the keys in this multimap, in the same order and 692 * with the same frequencies as they appear in this multimap; to get only a single occurrence of 693 * each key, use {@link #keySet}. 694 */ 695 @Override 696 public ImmutableMultiset<K> keys() { 697 return (ImmutableMultiset<K>) super.keys(); 698 } 699 700 @Override 701 ImmutableMultiset<K> createKeys() { 702 return new Keys(); 703 } 704 705 @SuppressWarnings("serial") // Uses writeReplace, not default serialization 706 @WeakOuter 707 class Keys extends ImmutableMultiset<K> { 708 @Override 709 public boolean contains(@CheckForNull Object object) { 710 return containsKey(object); 711 } 712 713 @Override 714 public int count(@CheckForNull Object element) { 715 Collection<V> values = map.get(element); 716 return (values == null) ? 0 : values.size(); 717 } 718 719 @Override 720 public ImmutableSet<K> elementSet() { 721 return keySet(); 722 } 723 724 @Override 725 public int size() { 726 return ImmutableMultimap.this.size(); 727 } 728 729 @Override 730 Multiset.Entry<K> getEntry(int index) { 731 Map.Entry<K, ? extends Collection<V>> entry = map.entrySet().asList().get(index); 732 return Multisets.immutableEntry(entry.getKey(), entry.getValue().size()); 733 } 734 735 @Override 736 boolean isPartialView() { 737 return true; 738 } 739 740 @GwtIncompatible 741 @J2ktIncompatible 742 @Override 743 Object writeReplace() { 744 return new KeysSerializedForm(ImmutableMultimap.this); 745 } 746 747 @GwtIncompatible 748 @J2ktIncompatible 749 private void readObject(ObjectInputStream stream) throws InvalidObjectException { 750 throw new InvalidObjectException("Use KeysSerializedForm"); 751 } 752 } 753 754 @GwtIncompatible 755 @J2ktIncompatible 756 private static final class KeysSerializedForm implements Serializable { 757 final ImmutableMultimap<?, ?> multimap; 758 759 KeysSerializedForm(ImmutableMultimap<?, ?> multimap) { 760 this.multimap = multimap; 761 } 762 763 Object readResolve() { 764 return multimap.keys(); 765 } 766 } 767 768 /** 769 * Returns an immutable collection of the values in this multimap. Its iterator traverses the 770 * values for the first key, the values for the second key, and so on. 771 */ 772 @Override 773 public ImmutableCollection<V> values() { 774 return (ImmutableCollection<V>) super.values(); 775 } 776 777 @Override 778 ImmutableCollection<V> createValues() { 779 return new Values<>(this); 780 } 781 782 @Override 783 UnmodifiableIterator<V> valueIterator() { 784 return new UnmodifiableIterator<V>() { 785 Iterator<? extends ImmutableCollection<V>> valueCollectionItr = map.values().iterator(); 786 Iterator<V> valueItr = Iterators.emptyIterator(); 787 788 @Override 789 public boolean hasNext() { 790 return valueItr.hasNext() || valueCollectionItr.hasNext(); 791 } 792 793 @Override 794 public V next() { 795 if (!valueItr.hasNext()) { 796 valueItr = valueCollectionItr.next().iterator(); 797 } 798 return valueItr.next(); 799 } 800 }; 801 } 802 803 private static final class Values<K, V> extends ImmutableCollection<V> { 804 @Weak private final transient ImmutableMultimap<K, V> multimap; 805 806 Values(ImmutableMultimap<K, V> multimap) { 807 this.multimap = multimap; 808 } 809 810 @Override 811 public boolean contains(@CheckForNull Object object) { 812 return multimap.containsValue(object); 813 } 814 815 @Override 816 public UnmodifiableIterator<V> iterator() { 817 return multimap.valueIterator(); 818 } 819 820 @GwtIncompatible // not present in emulated superclass 821 @Override 822 int copyIntoArray(@Nullable Object[] dst, int offset) { 823 for (ImmutableCollection<V> valueCollection : multimap.map.values()) { 824 offset = valueCollection.copyIntoArray(dst, offset); 825 } 826 return offset; 827 } 828 829 @Override 830 public int size() { 831 return multimap.size(); 832 } 833 834 @Override 835 boolean isPartialView() { 836 return true; 837 } 838 839 // redeclare to help optimizers with b/310253115 840 @SuppressWarnings("RedundantOverride") 841 @Override 842 @J2ktIncompatible // serialization 843 @GwtIncompatible // serialization 844 Object writeReplace() { 845 return super.writeReplace(); 846 } 847 848 @J2ktIncompatible // serialization 849 private static final long serialVersionUID = 0; 850 } 851 852 @J2ktIncompatible // serialization 853 private static final long serialVersionUID = 0; 854}