001/* 002 * Copyright (C) 2007 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.checkPositionIndex; 020import static com.google.common.base.Preconditions.checkState; 021import static com.google.common.collect.CollectPreconditions.checkRemove; 022import static java.util.Collections.unmodifiableList; 023 024import com.google.common.annotations.GwtCompatible; 025import com.google.common.annotations.GwtIncompatible; 026import com.google.j2objc.annotations.WeakOuter; 027 028import java.io.IOException; 029import java.io.ObjectInputStream; 030import java.io.ObjectOutputStream; 031import java.io.Serializable; 032import java.util.AbstractSequentialList; 033import java.util.Collection; 034import java.util.ConcurrentModificationException; 035import java.util.HashMap; 036import java.util.Iterator; 037import java.util.List; 038import java.util.ListIterator; 039import java.util.Map; 040import java.util.Map.Entry; 041import java.util.NoSuchElementException; 042import java.util.Set; 043 044import javax.annotation.Nullable; 045 046/** 047 * An implementation of {@code ListMultimap} that supports deterministic 048 * iteration order for both keys and values. The iteration order is preserved 049 * across non-distinct key values. For example, for the following multimap 050 * definition: <pre> {@code 051 * 052 * Multimap<K, V> multimap = LinkedListMultimap.create(); 053 * multimap.put(key1, foo); 054 * multimap.put(key2, bar); 055 * multimap.put(key1, baz);}</pre> 056 * 057 * ... the iteration order for {@link #keys()} is {@code [key1, key2, key1]}, 058 * and similarly for {@link #entries()}. Unlike {@link LinkedHashMultimap}, the 059 * iteration order is kept consistent between keys, entries and values. For 060 * example, calling: <pre> {@code 061 * 062 * map.remove(key1, foo);}</pre> 063 * 064 * <p>changes the entries iteration order to {@code [key2=bar, key1=baz]} and the 065 * key iteration order to {@code [key2, key1]}. The {@link #entries()} iterator 066 * returns mutable map entries, and {@link #replaceValues} attempts to preserve 067 * iteration order as much as possible. 068 * 069 * <p>The collections returned by {@link #keySet()} and {@link #asMap} iterate 070 * through the keys in the order they were first added to the multimap. 071 * Similarly, {@link #get}, {@link #removeAll}, and {@link #replaceValues} 072 * return collections that iterate through the values in the order they were 073 * added. The collections generated by {@link #entries()}, {@link #keys()}, and 074 * {@link #values} iterate across the key-value mappings in the order they were 075 * added to the multimap. 076 * 077 * <p>The {@link #values()} and {@link #entries()} methods both return a 078 * {@code List}, instead of the {@code Collection} specified by the {@link 079 * ListMultimap} interface. 080 * 081 * <p>The methods {@link #get}, {@link #keySet()}, {@link #keys()}, 082 * {@link #values}, {@link #entries()}, and {@link #asMap} return collections 083 * that are views of the multimap. If the multimap is modified while an 084 * iteration over any of those collections is in progress, except through the 085 * iterator's methods, the results of the iteration are undefined. 086 * 087 * <p>Keys and values may be null. All optional multimap methods are supported, 088 * and all returned views are modifiable. 089 * 090 * <p>This class is not threadsafe when any concurrent operations update the 091 * multimap. Concurrent read operations will work correctly. To allow concurrent 092 * update operations, wrap your multimap with a call to {@link 093 * Multimaps#synchronizedListMultimap}. 094 * 095 * <p>See the Guava User Guide article on <a href= 096 * "https://github.com/google/guava/wiki/NewCollectionTypesExplained#multimap"> 097 * {@code Multimap}</a>. 098 * 099 * @author Mike Bostock 100 * @since 2.0 101 */ 102@GwtCompatible(serializable = true, emulated = true) 103public class LinkedListMultimap<K, V> extends AbstractMultimap<K, V> 104 implements ListMultimap<K, V>, Serializable { 105 /* 106 * Order is maintained using a linked list containing all key-value pairs. In 107 * addition, a series of disjoint linked lists of "siblings", each containing 108 * the values for a specific key, is used to implement {@link 109 * ValueForKeyIterator} in constant time. 110 */ 111 112 private static final class Node<K, V> extends AbstractMapEntry<K, V> { 113 final K key; 114 V value; 115 Node<K, V> next; // the next node (with any key) 116 Node<K, V> previous; // the previous node (with any key) 117 Node<K, V> nextSibling; // the next node with the same key 118 Node<K, V> previousSibling; // the previous node with the same key 119 120 Node(@Nullable K key, @Nullable V value) { 121 this.key = key; 122 this.value = value; 123 } 124 125 @Override 126 public K getKey() { 127 return key; 128 } 129 130 @Override 131 public V getValue() { 132 return value; 133 } 134 135 @Override 136 public V setValue(@Nullable V newValue) { 137 V result = value; 138 this.value = newValue; 139 return result; 140 } 141 } 142 143 private static class KeyList<K, V> { 144 Node<K, V> head; 145 Node<K, V> tail; 146 int count; 147 148 KeyList(Node<K, V> firstNode) { 149 this.head = firstNode; 150 this.tail = firstNode; 151 firstNode.previousSibling = null; 152 firstNode.nextSibling = null; 153 this.count = 1; 154 } 155 } 156 157 private transient Node<K, V> head; // the head for all keys 158 private transient Node<K, V> tail; // the tail for all keys 159 private transient Map<K, KeyList<K, V>> keyToKeyList; 160 private transient int size; 161 162 /* 163 * Tracks modifications to keyToKeyList so that addition or removal of keys invalidates 164 * preexisting iterators. This does *not* track simple additions and removals of values 165 * that are not the first to be added or last to be removed for their key. 166 */ 167 private transient int modCount; 168 169 /** 170 * Creates a new, empty {@code LinkedListMultimap} with the default initial 171 * capacity. 172 */ 173 public static <K, V> LinkedListMultimap<K, V> create() { 174 return new LinkedListMultimap<K, V>(); 175 } 176 177 /** 178 * Constructs an empty {@code LinkedListMultimap} with enough capacity to hold 179 * the specified number of keys without rehashing. 180 * 181 * @param expectedKeys the expected number of distinct keys 182 * @throws IllegalArgumentException if {@code expectedKeys} is negative 183 */ 184 public static <K, V> LinkedListMultimap<K, V> create(int expectedKeys) { 185 return new LinkedListMultimap<K, V>(expectedKeys); 186 } 187 188 /** 189 * Constructs a {@code LinkedListMultimap} with the same mappings as the 190 * specified {@code Multimap}. The new multimap has the same 191 * {@link Multimap#entries()} iteration order as the input multimap. 192 * 193 * @param multimap the multimap whose contents are copied to this multimap 194 */ 195 public static <K, V> LinkedListMultimap<K, V> create( 196 Multimap<? extends K, ? extends V> multimap) { 197 return new LinkedListMultimap<K, V>(multimap); 198 } 199 200 LinkedListMultimap() { 201 keyToKeyList = Maps.newHashMap(); 202 } 203 204 private LinkedListMultimap(int expectedKeys) { 205 keyToKeyList = new HashMap<K, KeyList<K, V>>(expectedKeys); 206 } 207 208 private LinkedListMultimap(Multimap<? extends K, ? extends V> multimap) { 209 this(multimap.keySet().size()); 210 putAll(multimap); 211 } 212 213 /** 214 * Adds a new node for the specified key-value pair before the specified 215 * {@code nextSibling} element, or at the end of the list if {@code 216 * nextSibling} is null. Note: if {@code nextSibling} is specified, it MUST be 217 * for an node for the same {@code key}! 218 */ 219 private Node<K, V> addNode( 220 @Nullable K key, @Nullable V value, @Nullable Node<K, V> nextSibling) { 221 Node<K, V> node = new Node<K, V>(key, value); 222 if (head == null) { // empty list 223 head = tail = node; 224 keyToKeyList.put(key, new KeyList<K, V>(node)); 225 modCount++; 226 } else if (nextSibling == null) { // non-empty list, add to tail 227 tail.next = node; 228 node.previous = tail; 229 tail = node; 230 KeyList<K, V> keyList = keyToKeyList.get(key); 231 if (keyList == null) { 232 keyToKeyList.put(key, keyList = new KeyList<K, V>(node)); 233 modCount++; 234 } else { 235 keyList.count++; 236 Node<K, V> keyTail = keyList.tail; 237 keyTail.nextSibling = node; 238 node.previousSibling = keyTail; 239 keyList.tail = node; 240 } 241 } else { // non-empty list, insert before nextSibling 242 KeyList<K, V> keyList = keyToKeyList.get(key); 243 keyList.count++; 244 node.previous = nextSibling.previous; 245 node.previousSibling = nextSibling.previousSibling; 246 node.next = nextSibling; 247 node.nextSibling = nextSibling; 248 if (nextSibling.previousSibling == null) { // nextSibling was key head 249 keyToKeyList.get(key).head = node; 250 } else { 251 nextSibling.previousSibling.nextSibling = node; 252 } 253 if (nextSibling.previous == null) { // nextSibling was head 254 head = node; 255 } else { 256 nextSibling.previous.next = node; 257 } 258 nextSibling.previous = node; 259 nextSibling.previousSibling = node; 260 } 261 size++; 262 return node; 263 } 264 265 /** 266 * Removes the specified node from the linked list. This method is only 267 * intended to be used from the {@code Iterator} classes. See also {@link 268 * LinkedListMultimap#removeAllNodes(Object)}. 269 */ 270 private void removeNode(Node<K, V> node) { 271 if (node.previous != null) { 272 node.previous.next = node.next; 273 } else { // node was head 274 head = node.next; 275 } 276 if (node.next != null) { 277 node.next.previous = node.previous; 278 } else { // node was tail 279 tail = node.previous; 280 } 281 if (node.previousSibling == null && node.nextSibling == null) { 282 KeyList<K, V> keyList = keyToKeyList.remove(node.key); 283 keyList.count = 0; 284 modCount++; 285 } else { 286 KeyList<K, V> keyList = keyToKeyList.get(node.key); 287 keyList.count--; 288 289 if (node.previousSibling == null) { 290 keyList.head = node.nextSibling; 291 } else { 292 node.previousSibling.nextSibling = node.nextSibling; 293 } 294 295 if (node.nextSibling == null) { 296 keyList.tail = node.previousSibling; 297 } else { 298 node.nextSibling.previousSibling = node.previousSibling; 299 } 300 } 301 size--; 302 } 303 304 /** Removes all nodes for the specified key. */ 305 private void removeAllNodes(@Nullable Object key) { 306 Iterators.clear(new ValueForKeyIterator(key)); 307 } 308 309 /** Helper method for verifying that an iterator element is present. */ 310 private static void checkElement(@Nullable Object node) { 311 if (node == null) { 312 throw new NoSuchElementException(); 313 } 314 } 315 316 /** An {@code Iterator} over all nodes. */ 317 private class NodeIterator implements ListIterator<Entry<K, V>> { 318 int nextIndex; 319 Node<K, V> next; 320 Node<K, V> current; 321 Node<K, V> previous; 322 int expectedModCount = modCount; 323 324 NodeIterator(int index) { 325 int size = size(); 326 checkPositionIndex(index, size); 327 if (index >= (size / 2)) { 328 previous = tail; 329 nextIndex = size; 330 while (index++ < size) { 331 previous(); 332 } 333 } else { 334 next = head; 335 while (index-- > 0) { 336 next(); 337 } 338 } 339 current = null; 340 } 341 private void checkForConcurrentModification() { 342 if (modCount != expectedModCount) { 343 throw new ConcurrentModificationException(); 344 } 345 } 346 @Override 347 public boolean hasNext() { 348 checkForConcurrentModification(); 349 return next != null; 350 } 351 @Override 352 public Node<K, V> next() { 353 checkForConcurrentModification(); 354 checkElement(next); 355 previous = current = next; 356 next = next.next; 357 nextIndex++; 358 return current; 359 } 360 @Override 361 public void remove() { 362 checkForConcurrentModification(); 363 checkRemove(current != null); 364 if (current != next) { // after call to next() 365 previous = current.previous; 366 nextIndex--; 367 } else { // after call to previous() 368 next = current.next; 369 } 370 removeNode(current); 371 current = null; 372 expectedModCount = modCount; 373 } 374 @Override 375 public boolean hasPrevious() { 376 checkForConcurrentModification(); 377 return previous != null; 378 } 379 @Override 380 public Node<K, V> previous() { 381 checkForConcurrentModification(); 382 checkElement(previous); 383 next = current = previous; 384 previous = previous.previous; 385 nextIndex--; 386 return current; 387 } 388 @Override 389 public int nextIndex() { 390 return nextIndex; 391 } 392 @Override 393 public int previousIndex() { 394 return nextIndex - 1; 395 } 396 @Override 397 public void set(Entry<K, V> e) { 398 throw new UnsupportedOperationException(); 399 } 400 @Override 401 public void add(Entry<K, V> e) { 402 throw new UnsupportedOperationException(); 403 } 404 void setValue(V value) { 405 checkState(current != null); 406 current.value = value; 407 } 408 } 409 410 /** An {@code Iterator} over distinct keys in key head order. */ 411 private class DistinctKeyIterator implements Iterator<K> { 412 final Set<K> seenKeys = Sets.<K>newHashSetWithExpectedSize(keySet().size()); 413 Node<K, V> next = head; 414 Node<K, V> current; 415 int expectedModCount = modCount; 416 417 private void checkForConcurrentModification() { 418 if (modCount != expectedModCount) { 419 throw new ConcurrentModificationException(); 420 } 421 } 422 @Override 423 public boolean hasNext() { 424 checkForConcurrentModification(); 425 return next != null; 426 } 427 @Override 428 public K next() { 429 checkForConcurrentModification(); 430 checkElement(next); 431 current = next; 432 seenKeys.add(current.key); 433 do { // skip ahead to next unseen key 434 next = next.next; 435 } while ((next != null) && !seenKeys.add(next.key)); 436 return current.key; 437 } 438 @Override 439 public void remove() { 440 checkForConcurrentModification(); 441 checkRemove(current != null); 442 removeAllNodes(current.key); 443 current = null; 444 expectedModCount = modCount; 445 } 446 } 447 448 /** A {@code ListIterator} over values for a specified key. */ 449 private class ValueForKeyIterator implements ListIterator<V> { 450 final Object key; 451 int nextIndex; 452 Node<K, V> next; 453 Node<K, V> current; 454 Node<K, V> previous; 455 456 /** Constructs a new iterator over all values for the specified key. */ 457 ValueForKeyIterator(@Nullable Object key) { 458 this.key = key; 459 KeyList<K, V> keyList = keyToKeyList.get(key); 460 next = (keyList == null) ? null : keyList.head; 461 } 462 463 /** 464 * Constructs a new iterator over all values for the specified key starting 465 * at the specified index. This constructor is optimized so that it starts 466 * at either the head or the tail, depending on which is closer to the 467 * specified index. This allows adds to the tail to be done in constant 468 * time. 469 * 470 * @throws IndexOutOfBoundsException if index is invalid 471 */ 472 public ValueForKeyIterator(@Nullable Object key, int index) { 473 KeyList<K, V> keyList = keyToKeyList.get(key); 474 int size = (keyList == null) ? 0 : keyList.count; 475 checkPositionIndex(index, size); 476 if (index >= (size / 2)) { 477 previous = (keyList == null) ? null : keyList.tail; 478 nextIndex = size; 479 while (index++ < size) { 480 previous(); 481 } 482 } else { 483 next = (keyList == null) ? null : keyList.head; 484 while (index-- > 0) { 485 next(); 486 } 487 } 488 this.key = key; 489 current = null; 490 } 491 492 @Override 493 public boolean hasNext() { 494 return next != null; 495 } 496 497 @Override 498 public V next() { 499 checkElement(next); 500 previous = current = next; 501 next = next.nextSibling; 502 nextIndex++; 503 return current.value; 504 } 505 506 @Override 507 public boolean hasPrevious() { 508 return previous != null; 509 } 510 511 @Override 512 public V previous() { 513 checkElement(previous); 514 next = current = previous; 515 previous = previous.previousSibling; 516 nextIndex--; 517 return current.value; 518 } 519 520 @Override 521 public int nextIndex() { 522 return nextIndex; 523 } 524 525 @Override 526 public int previousIndex() { 527 return nextIndex - 1; 528 } 529 530 @Override 531 public void remove() { 532 checkRemove(current != null); 533 if (current != next) { // after call to next() 534 previous = current.previousSibling; 535 nextIndex--; 536 } else { // after call to previous() 537 next = current.nextSibling; 538 } 539 removeNode(current); 540 current = null; 541 } 542 543 @Override 544 public void set(V value) { 545 checkState(current != null); 546 current.value = value; 547 } 548 549 @Override 550 @SuppressWarnings("unchecked") 551 public void add(V value) { 552 previous = addNode((K) key, value, next); 553 nextIndex++; 554 current = null; 555 } 556 } 557 558 // Query Operations 559 560 @Override 561 public int size() { 562 return size; 563 } 564 565 @Override 566 public boolean isEmpty() { 567 return head == null; 568 } 569 570 @Override 571 public boolean containsKey(@Nullable Object key) { 572 return keyToKeyList.containsKey(key); 573 } 574 575 @Override 576 public boolean containsValue(@Nullable Object value) { 577 return values().contains(value); 578 } 579 580 // Modification Operations 581 582 /** 583 * Stores a key-value pair in the multimap. 584 * 585 * @param key key to store in the multimap 586 * @param value value to store in the multimap 587 * @return {@code true} always 588 */ 589 @Override 590 public boolean put(@Nullable K key, @Nullable V value) { 591 addNode(key, value, null); 592 return true; 593 } 594 595 // Bulk Operations 596 597 /** 598 * {@inheritDoc} 599 * 600 * <p>If any entries for the specified {@code key} already exist in the 601 * multimap, their values are changed in-place without affecting the iteration 602 * order. 603 * 604 * <p>The returned list is immutable and implements 605 * {@link java.util.RandomAccess}. 606 */ 607 @Override 608 public List<V> replaceValues(@Nullable K key, Iterable<? extends V> values) { 609 List<V> oldValues = getCopy(key); 610 ListIterator<V> keyValues = new ValueForKeyIterator(key); 611 Iterator<? extends V> newValues = values.iterator(); 612 613 // Replace existing values, if any. 614 while (keyValues.hasNext() && newValues.hasNext()) { 615 keyValues.next(); 616 keyValues.set(newValues.next()); 617 } 618 619 // Remove remaining old values, if any. 620 while (keyValues.hasNext()) { 621 keyValues.next(); 622 keyValues.remove(); 623 } 624 625 // Add remaining new values, if any. 626 while (newValues.hasNext()) { 627 keyValues.add(newValues.next()); 628 } 629 630 return oldValues; 631 } 632 633 private List<V> getCopy(@Nullable Object key) { 634 return unmodifiableList(Lists.newArrayList(new ValueForKeyIterator(key))); 635 } 636 637 /** 638 * {@inheritDoc} 639 * 640 * <p>The returned list is immutable and implements 641 * {@link java.util.RandomAccess}. 642 */ 643 @Override 644 public List<V> removeAll(@Nullable Object key) { 645 List<V> oldValues = getCopy(key); 646 removeAllNodes(key); 647 return oldValues; 648 } 649 650 @Override 651 public void clear() { 652 head = null; 653 tail = null; 654 keyToKeyList.clear(); 655 size = 0; 656 modCount++; 657 } 658 659 // Views 660 661 /** 662 * {@inheritDoc} 663 * 664 * <p>If the multimap is modified while an iteration over the list is in 665 * progress (except through the iterator's own {@code add}, {@code set} or 666 * {@code remove} operations) the results of the iteration are undefined. 667 * 668 * <p>The returned list is not serializable and does not have random access. 669 */ 670 @Override 671 public List<V> get(final @Nullable K key) { 672 return new AbstractSequentialList<V>() { 673 @Override public int size() { 674 KeyList<K, V> keyList = keyToKeyList.get(key); 675 return (keyList == null) ? 0 : keyList.count; 676 } 677 @Override public ListIterator<V> listIterator(int index) { 678 return new ValueForKeyIterator(key, index); 679 } 680 }; 681 } 682 683 @Override 684 Set<K> createKeySet() { 685 @WeakOuter 686 class KeySetImpl extends Sets.ImprovedAbstractSet<K> { 687 @Override public int size() { 688 return keyToKeyList.size(); 689 } 690 @Override public Iterator<K> iterator() { 691 return new DistinctKeyIterator(); 692 } 693 @Override public boolean contains(Object key) { // for performance 694 return containsKey(key); 695 } 696 @Override 697 public boolean remove(Object o) { // for performance 698 return !LinkedListMultimap.this.removeAll(o).isEmpty(); 699 } 700 } 701 return new KeySetImpl(); 702 } 703 704 /** 705 * {@inheritDoc} 706 * 707 * <p>The iterator generated by the returned collection traverses the values 708 * in the order they were added to the multimap. Because the values may have 709 * duplicates and follow the insertion ordering, this method returns a {@link 710 * List}, instead of the {@link Collection} specified in the {@link 711 * ListMultimap} interface. 712 */ 713 @Override 714 public List<V> values() { 715 return (List<V>) super.values(); 716 } 717 718 @Override 719 List<V> createValues() { 720 @WeakOuter 721 class ValuesImpl extends AbstractSequentialList<V> { 722 @Override public int size() { 723 return size; 724 } 725 726 @Override public ListIterator<V> listIterator(int index) { 727 final NodeIterator nodeItr = new NodeIterator(index); 728 return new TransformedListIterator<Entry<K, V>, V>(nodeItr) { 729 @Override 730 V transform(Entry<K, V> entry) { 731 return entry.getValue(); 732 } 733 734 @Override 735 public void set(V value) { 736 nodeItr.setValue(value); 737 } 738 }; 739 } 740 } 741 return new ValuesImpl(); 742 } 743 744 /** 745 * {@inheritDoc} 746 * 747 * <p>The iterator generated by the returned collection traverses the entries 748 * in the order they were added to the multimap. Because the entries may have 749 * duplicates and follow the insertion ordering, this method returns a {@link 750 * List}, instead of the {@link Collection} specified in the {@link 751 * ListMultimap} interface. 752 * 753 * <p>An entry's {@link Entry#getKey} method always returns the same key, 754 * regardless of what happens subsequently. As long as the corresponding 755 * key-value mapping is not removed from the multimap, {@link Entry#getValue} 756 * returns the value from the multimap, which may change over time, and {@link 757 * Entry#setValue} modifies that value. Removing the mapping from the 758 * multimap does not alter the value returned by {@code getValue()}, though a 759 * subsequent {@code setValue()} call won't update the multimap but will lead 760 * to a revised value being returned by {@code getValue()}. 761 */ 762 @Override 763 public List<Entry<K, V>> entries() { 764 return (List<Entry<K, V>>) super.entries(); 765 } 766 767 @Override 768 List<Entry<K, V>> createEntries() { 769 @WeakOuter 770 class EntriesImpl extends AbstractSequentialList<Entry<K, V>> { 771 @Override public int size() { 772 return size; 773 } 774 775 @Override public ListIterator<Entry<K, V>> listIterator(int index) { 776 return new NodeIterator(index); 777 } 778 } 779 return new EntriesImpl(); 780 } 781 782 @Override 783 Iterator<Entry<K, V>> entryIterator() { 784 throw new AssertionError("should never be called"); 785 } 786 787 @Override 788 Map<K, Collection<V>> createAsMap() { 789 return new Multimaps.AsMap<K, V>(this); 790 } 791 792 /** 793 * @serialData the number of distinct keys, and then for each distinct key: 794 * the first key, the number of values for that key, and the key's values, 795 * followed by successive keys and values from the entries() ordering 796 */ 797 @GwtIncompatible("java.io.ObjectOutputStream") 798 private void writeObject(ObjectOutputStream stream) throws IOException { 799 stream.defaultWriteObject(); 800 stream.writeInt(size()); 801 for (Entry<K, V> entry : entries()) { 802 stream.writeObject(entry.getKey()); 803 stream.writeObject(entry.getValue()); 804 } 805 } 806 807 @GwtIncompatible("java.io.ObjectInputStream") 808 private void readObject(ObjectInputStream stream) 809 throws IOException, ClassNotFoundException { 810 stream.defaultReadObject(); 811 keyToKeyList = Maps.newLinkedHashMap(); 812 int size = stream.readInt(); 813 for (int i = 0; i < size; i++) { 814 @SuppressWarnings("unchecked") // reading data stored by writeObject 815 K key = (K) stream.readObject(); 816 @SuppressWarnings("unchecked") // reading data stored by writeObject 817 V value = (V) stream.readObject(); 818 put(key, value); 819 } 820 } 821 822 @GwtIncompatible("java serialization not supported") 823 private static final long serialVersionUID = 0; 824}