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.collect.CollectPreconditions.checkNonnegative; 020import static com.google.common.collect.CollectPreconditions.checkRemove; 021 022import com.google.common.annotations.GwtCompatible; 023import com.google.common.annotations.GwtIncompatible; 024import com.google.common.annotations.VisibleForTesting; 025import com.google.common.base.Objects; 026import com.google.j2objc.annotations.WeakOuter; 027 028import java.io.IOException; 029import java.io.ObjectInputStream; 030import java.io.ObjectOutputStream; 031import java.util.Arrays; 032import java.util.Collection; 033import java.util.ConcurrentModificationException; 034import java.util.Iterator; 035import java.util.LinkedHashMap; 036import java.util.LinkedHashSet; 037import java.util.Map; 038import java.util.NoSuchElementException; 039import java.util.Set; 040 041import javax.annotation.Nullable; 042 043/** 044 * Implementation of {@code Multimap} that does not allow duplicate key-value 045 * entries and that returns collections whose iterators follow the ordering in 046 * which the data was added to the multimap. 047 * 048 * <p>The collections returned by {@code keySet}, {@code keys}, and {@code 049 * asMap} iterate through the keys in the order they were first added to the 050 * multimap. Similarly, {@code get}, {@code removeAll}, and {@code 051 * replaceValues} return collections that iterate through the values in the 052 * order they were added. The collections generated by {@code entries} and 053 * {@code values} iterate across the key-value mappings in the order they were 054 * added to the multimap. 055 * 056 * <p>The iteration ordering of the collections generated by {@code keySet}, 057 * {@code keys}, and {@code asMap} has a few subtleties. As long as the set of 058 * keys remains unchanged, adding or removing mappings does not affect the key 059 * iteration order. However, if you remove all values associated with a key and 060 * then add the key back to the multimap, that key will come last in the key 061 * iteration order. 062 * 063 * <p>The multimap does not store duplicate key-value pairs. Adding a new 064 * key-value pair equal to an existing key-value pair has no effect. 065 * 066 * <p>Keys and values may be null. All optional multimap methods are supported, 067 * and all returned views are modifiable. 068 * 069 * <p>This class is not threadsafe when any concurrent operations update the 070 * multimap. Concurrent read operations will work correctly. To allow concurrent 071 * update operations, wrap your multimap with a call to {@link 072 * Multimaps#synchronizedSetMultimap}. 073 * 074 * <p>See the Guava User Guide article on <a href= 075 * "https://github.com/google/guava/wiki/NewCollectionTypesExplained#multimap"> 076 * {@code Multimap}</a>. 077 * 078 * @author Jared Levy 079 * @author Louis Wasserman 080 * @since 2.0 081 */ 082@GwtCompatible(serializable = true, emulated = true) 083public final class LinkedHashMultimap<K, V> extends AbstractSetMultimap<K, V> { 084 085 /** 086 * Creates a new, empty {@code LinkedHashMultimap} with the default initial 087 * capacities. 088 */ 089 public static <K, V> LinkedHashMultimap<K, V> create() { 090 return new LinkedHashMultimap<K, V>(DEFAULT_KEY_CAPACITY, DEFAULT_VALUE_SET_CAPACITY); 091 } 092 093 /** 094 * Constructs an empty {@code LinkedHashMultimap} with enough capacity to hold 095 * the specified numbers of keys and values without rehashing. 096 * 097 * @param expectedKeys the expected number of distinct keys 098 * @param expectedValuesPerKey the expected average number of values per key 099 * @throws IllegalArgumentException if {@code expectedKeys} or {@code 100 * expectedValuesPerKey} is negative 101 */ 102 public static <K, V> LinkedHashMultimap<K, V> create( 103 int expectedKeys, int expectedValuesPerKey) { 104 return new LinkedHashMultimap<K, V>( 105 Maps.capacity(expectedKeys), 106 Maps.capacity(expectedValuesPerKey)); 107 } 108 109 /** 110 * Constructs a {@code LinkedHashMultimap} with the same mappings as the 111 * specified multimap. If a key-value mapping appears multiple times in the 112 * input multimap, it only appears once in the constructed multimap. The new 113 * multimap has the same {@link Multimap#entries()} iteration order as the 114 * input multimap, except for excluding duplicate mappings. 115 * 116 * @param multimap the multimap whose contents are copied to this multimap 117 */ 118 public static <K, V> LinkedHashMultimap<K, V> create( 119 Multimap<? extends K, ? extends V> multimap) { 120 LinkedHashMultimap<K, V> result = create(multimap.keySet().size(), DEFAULT_VALUE_SET_CAPACITY); 121 result.putAll(multimap); 122 return result; 123 } 124 125 private interface ValueSetLink<K, V> { 126 ValueSetLink<K, V> getPredecessorInValueSet(); 127 ValueSetLink<K, V> getSuccessorInValueSet(); 128 129 void setPredecessorInValueSet(ValueSetLink<K, V> entry); 130 void setSuccessorInValueSet(ValueSetLink<K, V> entry); 131 } 132 133 private static <K, V> void succeedsInValueSet(ValueSetLink<K, V> pred, ValueSetLink<K, V> succ) { 134 pred.setSuccessorInValueSet(succ); 135 succ.setPredecessorInValueSet(pred); 136 } 137 138 private static <K, V> void succeedsInMultimap( 139 ValueEntry<K, V> pred, ValueEntry<K, V> succ) { 140 pred.setSuccessorInMultimap(succ); 141 succ.setPredecessorInMultimap(pred); 142 } 143 144 private static <K, V> void deleteFromValueSet(ValueSetLink<K, V> entry) { 145 succeedsInValueSet(entry.getPredecessorInValueSet(), entry.getSuccessorInValueSet()); 146 } 147 148 private static <K, V> void deleteFromMultimap(ValueEntry<K, V> entry) { 149 succeedsInMultimap(entry.getPredecessorInMultimap(), entry.getSuccessorInMultimap()); 150 } 151 152 /** 153 * LinkedHashMultimap entries are in no less than three coexisting linked lists: 154 * a bucket in the hash table for a Set<V> associated with a key, the linked list 155 * of insertion-ordered entries in that Set<V>, and the linked list of entries 156 * in the LinkedHashMultimap as a whole. 157 */ 158 @VisibleForTesting 159 static final class ValueEntry<K, V> extends ImmutableEntry<K, V> 160 implements ValueSetLink<K, V> { 161 final int smearedValueHash; 162 163 @Nullable ValueEntry<K, V> nextInValueBucket; 164 165 ValueSetLink<K, V> predecessorInValueSet; 166 ValueSetLink<K, V> successorInValueSet; 167 168 ValueEntry<K, V> predecessorInMultimap; 169 ValueEntry<K, V> successorInMultimap; 170 171 ValueEntry(@Nullable K key, @Nullable V value, int smearedValueHash, 172 @Nullable ValueEntry<K, V> nextInValueBucket) { 173 super(key, value); 174 this.smearedValueHash = smearedValueHash; 175 this.nextInValueBucket = nextInValueBucket; 176 } 177 178 boolean matchesValue(@Nullable Object v, int smearedVHash) { 179 return smearedValueHash == smearedVHash && Objects.equal(getValue(), v); 180 } 181 182 @Override 183 public ValueSetLink<K, V> getPredecessorInValueSet() { 184 return predecessorInValueSet; 185 } 186 187 @Override 188 public ValueSetLink<K, V> getSuccessorInValueSet() { 189 return successorInValueSet; 190 } 191 192 @Override 193 public void setPredecessorInValueSet(ValueSetLink<K, V> entry) { 194 predecessorInValueSet = entry; 195 } 196 197 @Override 198 public void setSuccessorInValueSet(ValueSetLink<K, V> entry) { 199 successorInValueSet = entry; 200 } 201 202 public ValueEntry<K, V> getPredecessorInMultimap() { 203 return predecessorInMultimap; 204 } 205 206 public ValueEntry<K, V> getSuccessorInMultimap() { 207 return successorInMultimap; 208 } 209 210 public void setSuccessorInMultimap(ValueEntry<K, V> multimapSuccessor) { 211 this.successorInMultimap = multimapSuccessor; 212 } 213 214 public void setPredecessorInMultimap(ValueEntry<K, V> multimapPredecessor) { 215 this.predecessorInMultimap = multimapPredecessor; 216 } 217 } 218 219 private static final int DEFAULT_KEY_CAPACITY = 16; 220 private static final int DEFAULT_VALUE_SET_CAPACITY = 2; 221 @VisibleForTesting static final double VALUE_SET_LOAD_FACTOR = 1.0; 222 223 @VisibleForTesting transient int valueSetCapacity = DEFAULT_VALUE_SET_CAPACITY; 224 private transient ValueEntry<K, V> multimapHeaderEntry; 225 226 private LinkedHashMultimap(int keyCapacity, int valueSetCapacity) { 227 super(new LinkedHashMap<K, Collection<V>>(keyCapacity)); 228 checkNonnegative(valueSetCapacity, "expectedValuesPerKey"); 229 230 this.valueSetCapacity = valueSetCapacity; 231 this.multimapHeaderEntry = new ValueEntry<K, V>(null, null, 0, null); 232 succeedsInMultimap(multimapHeaderEntry, multimapHeaderEntry); 233 } 234 235 /** 236 * {@inheritDoc} 237 * 238 * <p>Creates an empty {@code LinkedHashSet} for a collection of values for 239 * one key. 240 * 241 * @return a new {@code LinkedHashSet} containing a collection of values for 242 * one key 243 */ 244 @Override 245 Set<V> createCollection() { 246 return new LinkedHashSet<V>(valueSetCapacity); 247 } 248 249 /** 250 * {@inheritDoc} 251 * 252 * <p>Creates a decorated insertion-ordered set that also keeps track of the 253 * order in which key-value pairs are added to the multimap. 254 * 255 * @param key key to associate with values in the collection 256 * @return a new decorated set containing a collection of values for one key 257 */ 258 @Override 259 Collection<V> createCollection(K key) { 260 return new ValueSet(key, valueSetCapacity); 261 } 262 263 /** 264 * {@inheritDoc} 265 * 266 * <p>If {@code values} is not empty and the multimap already contains a 267 * mapping for {@code key}, the {@code keySet()} ordering is unchanged. 268 * However, the provided values always come last in the {@link #entries()} and 269 * {@link #values()} iteration orderings. 270 */ 271 @Override 272 public Set<V> replaceValues(@Nullable K key, Iterable<? extends V> values) { 273 return super.replaceValues(key, values); 274 } 275 276 /** 277 * Returns a set of all key-value pairs. Changes to the returned set will 278 * update the underlying multimap, and vice versa. The entries set does not 279 * support the {@code add} or {@code addAll} operations. 280 * 281 * <p>The iterator generated by the returned set traverses the entries in the 282 * order they were added to the multimap. 283 * 284 * <p>Each entry is an immutable snapshot of a key-value mapping in the 285 * multimap, taken at the time the entry is returned by a method call to the 286 * collection or its iterator. 287 */ 288 @Override public Set<Map.Entry<K, V>> entries() { 289 return super.entries(); 290 } 291 292 /** 293 * Returns a collection of all values in the multimap. Changes to the returned 294 * collection will update the underlying multimap, and vice versa. 295 * 296 * <p>The iterator generated by the returned collection traverses the values 297 * in the order they were added to the multimap. 298 */ 299 @Override public Collection<V> values() { 300 return super.values(); 301 } 302 303 @VisibleForTesting 304 @WeakOuter 305 final class ValueSet extends Sets.ImprovedAbstractSet<V> implements ValueSetLink<K, V> { 306 /* 307 * We currently use a fixed load factor of 1.0, a bit higher than normal to reduce memory 308 * consumption. 309 */ 310 311 private final K key; 312 @VisibleForTesting ValueEntry<K, V>[] hashTable; 313 private int size = 0; 314 private int modCount = 0; 315 316 // We use the set object itself as the end of the linked list, avoiding an unnecessary 317 // entry object per key. 318 private ValueSetLink<K, V> firstEntry; 319 private ValueSetLink<K, V> lastEntry; 320 321 ValueSet(K key, int expectedValues) { 322 this.key = key; 323 this.firstEntry = this; 324 this.lastEntry = this; 325 // Round expected values up to a power of 2 to get the table size. 326 int tableSize = Hashing.closedTableSize(expectedValues, VALUE_SET_LOAD_FACTOR); 327 328 @SuppressWarnings("unchecked") 329 ValueEntry<K, V>[] hashTable = new ValueEntry[tableSize]; 330 this.hashTable = hashTable; 331 } 332 333 private int mask() { 334 return hashTable.length - 1; 335 } 336 337 @Override 338 public ValueSetLink<K, V> getPredecessorInValueSet() { 339 return lastEntry; 340 } 341 342 @Override 343 public ValueSetLink<K, V> getSuccessorInValueSet() { 344 return firstEntry; 345 } 346 347 @Override 348 public void setPredecessorInValueSet(ValueSetLink<K, V> entry) { 349 lastEntry = entry; 350 } 351 352 @Override 353 public void setSuccessorInValueSet(ValueSetLink<K, V> entry) { 354 firstEntry = entry; 355 } 356 357 @Override 358 public Iterator<V> iterator() { 359 return new Iterator<V>() { 360 ValueSetLink<K, V> nextEntry = firstEntry; 361 ValueEntry<K, V> toRemove; 362 int expectedModCount = modCount; 363 364 private void checkForComodification() { 365 if (modCount != expectedModCount) { 366 throw new ConcurrentModificationException(); 367 } 368 } 369 370 @Override 371 public boolean hasNext() { 372 checkForComodification(); 373 return nextEntry != ValueSet.this; 374 } 375 376 @Override 377 public V next() { 378 if (!hasNext()) { 379 throw new NoSuchElementException(); 380 } 381 ValueEntry<K, V> entry = (ValueEntry<K, V>) nextEntry; 382 V result = entry.getValue(); 383 toRemove = entry; 384 nextEntry = entry.getSuccessorInValueSet(); 385 return result; 386 } 387 388 @Override 389 public void remove() { 390 checkForComodification(); 391 checkRemove(toRemove != null); 392 ValueSet.this.remove(toRemove.getValue()); 393 expectedModCount = modCount; 394 toRemove = null; 395 } 396 }; 397 } 398 399 @Override 400 public int size() { 401 return size; 402 } 403 404 @Override 405 public boolean contains(@Nullable Object o) { 406 int smearedHash = Hashing.smearedHash(o); 407 for (ValueEntry<K, V> entry = hashTable[smearedHash & mask()]; entry != null; 408 entry = entry.nextInValueBucket) { 409 if (entry.matchesValue(o, smearedHash)) { 410 return true; 411 } 412 } 413 return false; 414 } 415 416 @Override 417 public boolean add(@Nullable V value) { 418 int smearedHash = Hashing.smearedHash(value); 419 int bucket = smearedHash & mask(); 420 ValueEntry<K, V> rowHead = hashTable[bucket]; 421 for (ValueEntry<K, V> entry = rowHead; entry != null; 422 entry = entry.nextInValueBucket) { 423 if (entry.matchesValue(value, smearedHash)) { 424 return false; 425 } 426 } 427 428 ValueEntry<K, V> newEntry = new ValueEntry<K, V>(key, value, smearedHash, rowHead); 429 succeedsInValueSet(lastEntry, newEntry); 430 succeedsInValueSet(newEntry, this); 431 succeedsInMultimap(multimapHeaderEntry.getPredecessorInMultimap(), newEntry); 432 succeedsInMultimap(newEntry, multimapHeaderEntry); 433 hashTable[bucket] = newEntry; 434 size++; 435 modCount++; 436 rehashIfNecessary(); 437 return true; 438 } 439 440 private void rehashIfNecessary() { 441 if (Hashing.needsResizing(size, hashTable.length, VALUE_SET_LOAD_FACTOR)) { 442 @SuppressWarnings("unchecked") 443 ValueEntry<K, V>[] hashTable = new ValueEntry[this.hashTable.length * 2]; 444 this.hashTable = hashTable; 445 int mask = hashTable.length - 1; 446 for (ValueSetLink<K, V> entry = firstEntry; 447 entry != this; entry = entry.getSuccessorInValueSet()) { 448 ValueEntry<K, V> valueEntry = (ValueEntry<K, V>) entry; 449 int bucket = valueEntry.smearedValueHash & mask; 450 valueEntry.nextInValueBucket = hashTable[bucket]; 451 hashTable[bucket] = valueEntry; 452 } 453 } 454 } 455 456 @Override 457 public boolean remove(@Nullable Object o) { 458 int smearedHash = Hashing.smearedHash(o); 459 int bucket = smearedHash & mask(); 460 ValueEntry<K, V> prev = null; 461 for (ValueEntry<K, V> entry = hashTable[bucket]; entry != null; 462 prev = entry, entry = entry.nextInValueBucket) { 463 if (entry.matchesValue(o, smearedHash)) { 464 if (prev == null) { 465 // first entry in the bucket 466 hashTable[bucket] = entry.nextInValueBucket; 467 } else { 468 prev.nextInValueBucket = entry.nextInValueBucket; 469 } 470 deleteFromValueSet(entry); 471 deleteFromMultimap(entry); 472 size--; 473 modCount++; 474 return true; 475 } 476 } 477 return false; 478 } 479 480 @Override 481 public void clear() { 482 Arrays.fill(hashTable, null); 483 size = 0; 484 for (ValueSetLink<K, V> entry = firstEntry; 485 entry != this; entry = entry.getSuccessorInValueSet()) { 486 ValueEntry<K, V> valueEntry = (ValueEntry<K, V>) entry; 487 deleteFromMultimap(valueEntry); 488 } 489 succeedsInValueSet(this, this); 490 modCount++; 491 } 492 } 493 494 @Override 495 Iterator<Map.Entry<K, V>> entryIterator() { 496 return new Iterator<Map.Entry<K, V>>() { 497 ValueEntry<K, V> nextEntry = multimapHeaderEntry.successorInMultimap; 498 ValueEntry<K, V> toRemove; 499 500 @Override 501 public boolean hasNext() { 502 return nextEntry != multimapHeaderEntry; 503 } 504 505 @Override 506 public Map.Entry<K, V> next() { 507 if (!hasNext()) { 508 throw new NoSuchElementException(); 509 } 510 ValueEntry<K, V> result = nextEntry; 511 toRemove = result; 512 nextEntry = nextEntry.successorInMultimap; 513 return result; 514 } 515 516 @Override 517 public void remove() { 518 checkRemove(toRemove != null); 519 LinkedHashMultimap.this.remove(toRemove.getKey(), toRemove.getValue()); 520 toRemove = null; 521 } 522 }; 523 } 524 525 @Override 526 Iterator<V> valueIterator() { 527 return Maps.valueIterator(entryIterator()); 528 } 529 530 @Override 531 public void clear() { 532 super.clear(); 533 succeedsInMultimap(multimapHeaderEntry, multimapHeaderEntry); 534 } 535 536 /** 537 * @serialData the expected values per key, the number of distinct keys, 538 * the number of entries, and the entries in order 539 */ 540 @GwtIncompatible("java.io.ObjectOutputStream") 541 private void writeObject(ObjectOutputStream stream) throws IOException { 542 stream.defaultWriteObject(); 543 stream.writeInt(keySet().size()); 544 for (K key : keySet()) { 545 stream.writeObject(key); 546 } 547 stream.writeInt(size()); 548 for (Map.Entry<K, V> entry : entries()) { 549 stream.writeObject(entry.getKey()); 550 stream.writeObject(entry.getValue()); 551 } 552 } 553 554 @GwtIncompatible("java.io.ObjectInputStream") 555 private void readObject(ObjectInputStream stream) 556 throws IOException, ClassNotFoundException { 557 stream.defaultReadObject(); 558 multimapHeaderEntry = new ValueEntry<K, V>(null, null, 0, null); 559 succeedsInMultimap(multimapHeaderEntry, multimapHeaderEntry); 560 valueSetCapacity = DEFAULT_VALUE_SET_CAPACITY; 561 int distinctKeys = stream.readInt(); 562 Map<K, Collection<V>> map = new LinkedHashMap<K, Collection<V>>(); 563 for (int i = 0; i < distinctKeys; i++) { 564 @SuppressWarnings("unchecked") 565 K key = (K) stream.readObject(); 566 map.put(key, createCollection(key)); 567 } 568 int entries = stream.readInt(); 569 for (int i = 0; i < entries; i++) { 570 @SuppressWarnings("unchecked") 571 K key = (K) stream.readObject(); 572 @SuppressWarnings("unchecked") 573 V value = (V) stream.readObject(); 574 map.get(key).add(value); 575 } 576 setMap(map); 577 } 578 579 @GwtIncompatible("java serialization not supported") 580 private static final long serialVersionUID = 1; 581}