28
28
import java .util .ListIterator ;
29
29
import java .util .Map ;
30
30
import java .util .Map .Entry ;
31
+ import java .util .Optional ;
31
32
import java .util .Set ;
32
33
import java .util .SortedMap ;
33
34
import java .util .SortedSet ;
@@ -61,12 +62,12 @@ protected Map<K, Collection<V>> rawData() {
61
62
62
63
protected abstract Collection <V > createUnmodifiableEmptyCollection ();
63
64
64
- @ SuppressWarnings ( {"unchecked" , "rawtypes" } )
65
65
protected Collection <V > createUnmodifiable ( Collection <V > original ) {
66
- if (original instanceof List ) {
67
- return Collections .unmodifiableList ((List )original );
68
- } else if (original instanceof Set ) {
69
- return Collections .unmodifiableSet ((Set )original );
66
+ if (original instanceof List <?>) {
67
+ return Collections .unmodifiableList ((List <V >) original );
68
+ }
69
+ if (original instanceof Set <?>) {
70
+ return Collections .unmodifiableSet ((Set <V >) original );
70
71
}
71
72
return Collections .unmodifiableCollection (original );
72
73
}
@@ -97,21 +98,18 @@ public boolean containsKey( K key ) {
97
98
98
99
@ Override
99
100
public boolean containsValue ( Object value ) {
100
- for (Collection <V > collection : data .values ()) {
101
- if (collection .contains (value )) return true ;
102
- }
103
- return false ;
101
+ return data .values ().stream ().anyMatch (c -> c .contains (value ));
104
102
}
105
103
106
104
@ Override
107
105
public boolean containsEntry ( Object key ,
108
106
Object value ) {
109
107
Collection <V > values = data .get (key );
110
- return values == null ? false : values .contains (value );
108
+ return values != null && values .contains (value );
111
109
}
112
110
113
111
@ Override
114
- public java . util . Collection <V > get ( K key ) {
112
+ public Collection <V > get ( K key ) {
115
113
Collection <V > collection = data .get (key );
116
114
if (collection == null ) {
117
115
collection = createCollection ();
@@ -130,12 +128,7 @@ public boolean put( K key,
130
128
}
131
129
132
130
protected Collection <V > getOrCreateCollection ( K key ) {
133
- Collection <V > values = data .get (key );
134
- if (values == null ) {
135
- values = createCollection ();
136
- data .put (key , values );
137
- }
138
- return values ;
131
+ return data .computeIfAbsent (key , k -> createCollection ());
139
132
}
140
133
141
134
@ Override
@@ -153,7 +146,7 @@ public boolean remove( K key,
153
146
}
154
147
155
148
@ Override
156
- public java . util . Collection <V > removeAll ( K key ) {
149
+ public Collection <V > removeAll ( K key ) {
157
150
Collection <V > values = data .remove (key );
158
151
if (values == null ) return createUnmodifiableEmptyCollection ();
159
152
Collection <V > copy = createCollection ();
@@ -216,9 +209,6 @@ protected Iterator<Map.Entry<K, V>> createEntryIterator() {
216
209
217
210
protected Collection <V > wrapCollection ( K key ,
218
211
Collection <V > values ) {
219
- // if (values instanceof Set) {
220
- // return new WrappedSet(key, (Set<V>)values);
221
- // } else
222
212
if (values instanceof List ) {
223
213
return wrapList (key , (List <V >)values );
224
214
}
@@ -305,7 +295,7 @@ protected final class EntryIterator implements Iterator<Map.Entry<K, V>> {
305
295
if (iter .hasNext ()) {
306
296
nextKey ();
307
297
} else {
308
- currentValuesIterator = new EmptyIterator <V >();
298
+ currentValuesIterator = new EmptyIterator <>();
309
299
}
310
300
}
311
301
@@ -326,7 +316,7 @@ public Entry<K, V> next() {
326
316
if (!currentValuesIterator .hasNext ()) {
327
317
nextKey ();
328
318
}
329
- return new ImmutableMapEntry <K , V >(currentKey , currentValuesIterator .next ());
319
+ return new ImmutableMapEntry <>(currentKey , currentValuesIterator .next ());
330
320
}
331
321
332
322
@ SuppressWarnings ( "synthetic-access" )
@@ -365,8 +355,8 @@ public boolean contains( Object o ) {
365
355
}
366
356
367
357
protected class WrappedCollection implements Collection <V > {
368
- private Collection <V > delegate ;
369
358
private final K key ;
359
+ private Collection <V > delegate ;
370
360
371
361
protected WrappedCollection ( K key ,
372
362
Collection <V > values ) {
@@ -403,22 +393,25 @@ protected final void addToMap() {
403
393
404
394
@ Override
405
395
public int size () {
406
- return delegate ().size ();
396
+ Collection <V > d = delegate ();
397
+ return d == null ? 0 : d .size ();
407
398
}
408
399
409
400
@ Override
410
401
public boolean isEmpty () {
411
- return delegate ().isEmpty ();
402
+ Collection <V > d = delegate ();
403
+ return d == null || d .isEmpty ();
412
404
}
413
405
414
406
@ Override
415
407
public boolean contains ( Object o ) {
416
- return delegate ().contains (o );
408
+ Collection <V > d = delegate ();
409
+ return d != null && d .contains (o );
417
410
}
418
411
419
412
@ Override
420
413
public Iterator <V > iterator () {
421
- return delegate (). iterator ();
414
+ return new DelegateIterator ();
422
415
}
423
416
424
417
@ Override
@@ -435,17 +428,20 @@ public <T> T[] toArray( T[] a ) {
435
428
436
429
@ Override
437
430
public boolean add ( V e ) {
438
- Collection <V > values = delegate ();
439
- boolean wasEmpty = values .isEmpty ();
440
- if (!values .add (e )) return false ;
441
- incrementSize (1 );
442
- if (wasEmpty ) addToMap ();
443
- return true ;
431
+ boolean wasEmpty = isEmpty ();
432
+ try {
433
+ return AbstractMultimap .this .put (key , e );
434
+ } finally {
435
+ if (wasEmpty ) {
436
+ delegate ();
437
+ }
438
+ }
444
439
}
445
440
446
441
@ Override
447
442
public boolean remove ( Object o ) {
448
- if (!delegate ().remove (o )) return false ;
443
+ Collection <V > d = delegate ();
444
+ if (d == null || !d .remove (o )) return false ;
449
445
decrementSize (1 );
450
446
removeIfEmpty ();
451
447
return true ;
@@ -459,54 +455,59 @@ public boolean containsAll( Collection<?> c ) {
459
455
@ Override
460
456
public boolean addAll ( Collection <? extends V > c ) {
461
457
if (c .isEmpty ()) return false ;
462
- Collection < V > delegate = delegate ( );
463
- int sizeBefore = delegate . size ();
464
- if (! delegate . addAll ( c )) return false ;
465
- incrementSize ( delegate . size () - sizeBefore ) ;
466
- if ( sizeBefore == 0 ) addToMap ( );
458
+ if ( isEmpty ()) getOrCreateCollection ( key );
459
+ Collection < V > d = delegate ();
460
+ int sizeBefore = d . size () ;
461
+ if (! d . addAll ( c )) return false ;
462
+ incrementSize ( d . size () - sizeBefore );
467
463
return true ;
468
464
}
469
465
470
466
@ Override
471
467
public boolean removeAll ( Collection <?> c ) {
472
- if (c .isEmpty ()) return false ;
473
- Collection <V > delegate = delegate ();
474
- int sizeBefore = delegate .size ();
475
- if (!delegate .removeAll (c )) return false ;
476
- incrementSize (delegate .size () - sizeBefore ); // will be negative
468
+ if (c .isEmpty () || isEmpty ()) {
469
+ return false ;
470
+ }
471
+ Collection <V > d = delegate ();
472
+ int sizeBefore = d .size ();
473
+ if (!d .removeAll (c )) return false ;
474
+ incrementSize (d .size () - sizeBefore ); // will be negative
477
475
removeIfEmpty ();
478
476
return true ;
479
477
}
480
478
481
479
@ Override
482
480
public boolean retainAll ( Collection <?> c ) {
483
- if (c .isEmpty ()) return false ;
484
- Collection <V > delegate = delegate ();
485
- int sizeBefore = delegate .size ();
486
- if (!delegate .retainAll (c )) return false ;
487
- int diff = delegate .size () - sizeBefore ;
481
+ if (isEmpty ()) {
482
+ return false ;
483
+ }
484
+ Collection <V > d = delegate ();
485
+ int sizeBefore = d .size ();
486
+ if (!d .retainAll (c )) return false ;
487
+ int diff = d .size () - sizeBefore ;
488
488
incrementSize (diff ); // will change correctly if diff is negative
489
489
return true ;
490
490
}
491
491
492
492
@ Override
493
493
public void clear () {
494
- Collection <V > delegate = delegate ();
495
- int sizeBefore = delegate .size ();
496
- delegate .clear ();
494
+ if (isEmpty ()) return ;
495
+ Collection <V > d = delegate ();
496
+ int sizeBefore = d .size ();
497
+ d .clear ();
497
498
decrementSize (sizeBefore );
498
499
removeIfEmpty ();
499
500
}
500
501
501
502
@ Override
502
503
public int hashCode () {
503
- return delegate ().hashCode ();
504
+ return isEmpty () ? Collections . EMPTY_SET . hashCode () : delegate ().hashCode ();
504
505
}
505
506
506
507
@ Override
507
508
public boolean equals ( Object obj ) {
508
- if ( obj == this ) return true ;
509
- return delegate ().equals (obj );
509
+ return obj == this
510
+ || Optional . ofNullable ( delegate ()). orElseGet ( AbstractMultimap . this :: createUnmodifiableEmptyCollection ).equals (obj );
510
511
}
511
512
512
513
@ Override
@@ -515,8 +516,8 @@ public String toString() {
515
516
}
516
517
517
518
protected class DelegateIterator implements Iterator <V > {
518
- private final Collection <V > source ;
519
519
private final Iterator <V > iterator ;
520
+ private Collection <V > source ;
520
521
521
522
protected DelegateIterator () {
522
523
source = delegate ();
@@ -550,6 +551,9 @@ public void remove() {
550
551
iterator .remove ();
551
552
decrementSize (1 );
552
553
removeIfEmpty ();
554
+ if (isEmpty ()) {
555
+ source = null ;
556
+ }
553
557
}
554
558
}
555
559
}
@@ -634,6 +638,16 @@ public List<V> subList( int fromIndex,
634
638
return wrapList (getKey (), valueSublist );
635
639
}
636
640
641
+ /**
642
+ * {@inheritDoc}
643
+ *
644
+ * @see org.modeshape.common.collection.AbstractMultimap.WrappedCollection#hashCode()
645
+ */
646
+ @ Override
647
+ public int hashCode () {
648
+ return isEmpty () ? Collections .EMPTY_LIST .hashCode () : delegate ().hashCode ();
649
+ }
650
+
637
651
protected class DelegateListIterator extends DelegateIterator implements ListIterator <V > {
638
652
639
653
protected DelegateListIterator () {
@@ -780,7 +794,7 @@ public Map.Entry<K, Collection<V>> next() {
780
794
Map .Entry <K , Collection <V >> currentEntry = delegateIterator .next ();
781
795
currentValues = currentEntry .getValue ();
782
796
K key = currentEntry .getKey ();
783
- return new ImmutableMapEntry <K , Collection < V > >(key , wrapCollection (key , currentValues ));
797
+ return new ImmutableMapEntry <>(key , wrapCollection (key , currentValues ));
784
798
}
785
799
786
800
@ Override
0 commit comments