001/* 002 * Licensed to the Apache Software Foundation (ASF) under one or more 003 * contributor license agreements. See the NOTICE file distributed with 004 * this work for additional information regarding copyright ownership. 005 * The ASF licenses this file to You under the Apache License, Version 2.0 006 * (the "License"); you may not use this file except in compliance with 007 * the License. You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 014 * See the License for the specific language governing permissions and 015 * limitations under the License. 016 */ 017package org.apache.commons.collections4; 018 019import java.util.AbstractSet; 020import java.util.Arrays; 021import java.util.Collection; 022import java.util.Collections; 023import java.util.HashSet; 024import java.util.IdentityHashMap; 025import java.util.Iterator; 026import java.util.NavigableSet; 027import java.util.Set; 028import java.util.SortedSet; 029import java.util.TreeSet; 030 031import org.apache.commons.collections4.set.ListOrderedSet; 032import org.apache.commons.collections4.set.PredicatedNavigableSet; 033import org.apache.commons.collections4.set.PredicatedSet; 034import org.apache.commons.collections4.set.PredicatedSortedSet; 035import org.apache.commons.collections4.set.TransformedNavigableSet; 036import org.apache.commons.collections4.set.TransformedSet; 037import org.apache.commons.collections4.set.TransformedSortedSet; 038import org.apache.commons.collections4.set.UnmodifiableNavigableSet; 039import org.apache.commons.collections4.set.UnmodifiableSet; 040import org.apache.commons.collections4.set.UnmodifiableSortedSet; 041 042/** 043 * Provides utility methods and decorators for 044 * {@link Set} and {@link SortedSet} instances. 045 * 046 * @since 2.1 047 */ 048public class SetUtils { 049 050 /** 051 * An unmodifiable <b>view</b> of a set that may be backed by other sets. 052 * <p> 053 * If the decorated sets change, this view will change as well. The contents 054 * of this view can be transferred to another instance via the {@link #copyInto(Set)} 055 * and {@link #toSet()} methods. 056 * 057 * @param <E> the element type 058 * @since 4.1 059 */ 060 public static abstract class SetView<E> extends AbstractSet<E> { 061 062 /** 063 * Copies the contents of this view into the provided set. 064 * 065 * @param <S> the set type 066 * @param set the set for copying the contents 067 */ 068 public <S extends Set<E>> void copyInto(final S set) { 069 CollectionUtils.addAll(set, this); 070 } 071 072 /** 073 * Return an iterator for this view; the returned iterator is 074 * not required to be unmodifiable. 075 * @return a new iterator for this view 076 */ 077 protected abstract Iterator<E> createIterator(); 078 079 @Override 080 public Iterator<E> iterator() { 081 return IteratorUtils.unmodifiableIterator(createIterator()); 082 } 083 084 @Override 085 public int size() { 086 return IteratorUtils.size(iterator()); 087 } 088 089 /** 090 * Returns a new set containing the contents of this view. 091 * 092 * @return a new set containing all elements of this view 093 */ 094 public Set<E> toSet() { 095 final Set<E> set = new HashSet<>(size()); 096 copyInto(set); 097 return set; 098 } 099 } 100 101 /** 102 * An empty unmodifiable sorted set. 103 * This is not provided in the JDK. 104 */ 105 @SuppressWarnings("rawtypes") 106 public static final SortedSet EMPTY_SORTED_SET = 107 UnmodifiableSortedSet.unmodifiableSortedSet(new TreeSet<>()); 108 109 /** 110 * Returns a unmodifiable <b>view</b> containing the difference of the given 111 * {@link Set}s, denoted by {@code a \ b} (or {@code a - b}). 112 * <p> 113 * The returned view contains all elements of {@code a} that are not a member 114 * of {@code b}. 115 * 116 * @param <E> the generic type that is able to represent the types contained 117 * in both input sets. 118 * @param a the set to subtract from, must not be null 119 * @param b the set to subtract, must not be null 120 * @return a view of the relative complement of of the two sets 121 * @since 4.1 122 */ 123 public static <E> SetView<E> difference(final Set<? extends E> a, final Set<? extends E> b) { 124 if (a == null || b == null) { 125 throw new NullPointerException("Sets must not be null."); 126 } 127 128 final Predicate<E> notContainedInB = new Predicate<E>() { 129 @Override 130 public boolean evaluate(final E object) { 131 return !b.contains(object); 132 } 133 }; 134 135 return new SetView<E>() { 136 @Override 137 public boolean contains(final Object o) { 138 return a.contains(o) && !b.contains(o); 139 } 140 141 @Override 142 public Iterator<E> createIterator() { 143 return IteratorUtils.filteredIterator(a.iterator(), notContainedInB); 144 } 145 }; 146 } 147 148 /** 149 * Returns a unmodifiable <b>view</b> of the symmetric difference of the given 150 * {@link Set}s. 151 * <p> 152 * The returned view contains all elements of {@code a} and {@code b} that are 153 * not a member of the other set. 154 * <p> 155 * This is equivalent to {@code union(difference(a, b), difference(b, a))}. 156 * 157 * @param <E> the generic type that is able to represent the types contained 158 * in both input sets. 159 * @param a the first set, must not be null 160 * @param b the second set, must not be null 161 * @return a view of the symmetric difference of the two sets 162 * @since 4.1 163 */ 164 public static <E> SetView<E> disjunction(final Set<? extends E> a, final Set<? extends E> b) { 165 if (a == null || b == null) { 166 throw new NullPointerException("Sets must not be null."); 167 } 168 169 final SetView<E> aMinusB = difference(a, b); 170 final SetView<E> bMinusA = difference(b, a); 171 172 return new SetView<E>() { 173 @Override 174 public boolean contains(final Object o) { 175 return a.contains(o) ^ b.contains(o); 176 } 177 178 @Override 179 public Iterator<E> createIterator() { 180 return IteratorUtils.chainedIterator(aMinusB.iterator(), bMinusA.iterator()); 181 } 182 183 @Override 184 public boolean isEmpty() { 185 return aMinusB.isEmpty() && bMinusA.isEmpty(); 186 } 187 188 @Override 189 public int size() { 190 return aMinusB.size() + bMinusA.size(); 191 } 192 }; 193 } 194 195 /** 196 * Returns an immutable empty set if the argument is <code>null</code>, 197 * or the argument itself otherwise. 198 * 199 * @param <T> the element type 200 * @param set the set, possibly <code>null</code> 201 * @return an empty set if the argument is <code>null</code> 202 */ 203 public static <T> Set<T> emptyIfNull(final Set<T> set) { 204 return set == null ? Collections.<T>emptySet() : set; 205 } 206 207 //----------------------------------------------------------------------- 208 209 /** 210 * Get a typed empty unmodifiable Set. 211 * @param <E> the element type 212 * @return an empty Set 213 */ 214 public static <E> Set<E> emptySet() { 215 return Collections.<E>emptySet(); 216 } 217 218 /** 219 * Get a typed empty unmodifiable sorted set. 220 * @param <E> the element type 221 * @return an empty sorted Set 222 */ 223 @SuppressWarnings("unchecked") // empty set is OK for any type 224 public static <E> SortedSet<E> emptySortedSet() { 225 return EMPTY_SORTED_SET; 226 } 227 228 /** 229 * Generates a hash code using the algorithm specified in 230 * {@link java.util.Set#hashCode()}. 231 * <p> 232 * This method is useful for implementing <code>Set</code> when you cannot 233 * extend AbstractSet. The method takes Collection instances to enable other 234 * collection types to use the Set implementation algorithm. 235 * 236 * @param <T> the element type 237 * @see java.util.Set#hashCode() 238 * @param set the set to calculate the hash code for, may be null 239 * @return the hash code 240 */ 241 public static <T> int hashCodeForSet(final Collection<T> set) { 242 if (set == null) { 243 return 0; 244 } 245 246 int hashCode = 0; 247 for (final T obj : set) { 248 if (obj != null) { 249 hashCode += obj.hashCode(); 250 } 251 } 252 return hashCode; 253 } 254 255 /** 256 * Create a set from the given items. If the passed var-args argument is {@code 257 * null}, then the method returns {@code null}. 258 * @param <E> the element type 259 * @param items the elements that make up the new set 260 * @return a set 261 * @since 4.3 262 */ 263 public static <E> HashSet<E> hashSet(final E... items) { 264 if (items == null) { 265 return null; 266 } 267 return new HashSet<>(Arrays.asList(items)); 268 } 269 270 /** 271 * Returns a unmodifiable <b>view</b> of the intersection of the given {@link Set}s. 272 * <p> 273 * The returned view contains all elements that are members of both input sets 274 * ({@code a} and {@code b}). 275 * 276 * @param <E> the generic type that is able to represent the types contained 277 * in both input sets. 278 * @param a the first set, must not be null 279 * @param b the second set, must not be null 280 * @return a view of the intersection of the two sets 281 * @since 4.1 282 */ 283 public static <E> SetView<E> intersection(final Set<? extends E> a, final Set<? extends E> b) { 284 if (a == null || b == null) { 285 throw new NullPointerException("Sets must not be null."); 286 } 287 288 final Predicate<E> containedInB = new Predicate<E>() { 289 @Override 290 public boolean evaluate(final E object) { 291 return b.contains(object); 292 } 293 }; 294 295 return new SetView<E>() { 296 @Override 297 public boolean contains(final Object o) { 298 return a.contains(o) && b.contains(o); 299 } 300 301 @Override 302 public Iterator<E> createIterator() { 303 return IteratorUtils.filteredIterator(a.iterator(), containedInB); 304 } 305 }; 306 } 307 308 /** 309 * Tests two sets for equality as per the <code>equals()</code> contract 310 * in {@link java.util.Set#equals(java.lang.Object)}. 311 * <p> 312 * This method is useful for implementing <code>Set</code> when you cannot 313 * extend AbstractSet. The method takes Collection instances to enable other 314 * collection types to use the Set implementation algorithm. 315 * <p> 316 * The relevant text (slightly paraphrased as this is a static method) is: 317 * <blockquote> 318 * <p>Two sets are considered equal if they have 319 * the same size, and every member of the first set is contained in 320 * the second. This ensures that the {@code equals} method works 321 * properly across different implementations of the {@code Set} 322 * interface.</p> 323 * 324 * <p> 325 * This implementation first checks if the two sets are the same object: 326 * if so it returns {@code true}. Then, it checks if the two sets are 327 * identical in size; if not, it returns false. If so, it returns 328 * {@code a.containsAll((Collection) b)}.</p> 329 * </blockquote> 330 * 331 * @see java.util.Set 332 * @param set1 the first set, may be null 333 * @param set2 the second set, may be null 334 * @return whether the sets are equal by value comparison 335 */ 336 public static boolean isEqualSet(final Collection<?> set1, final Collection<?> set2) { 337 if (set1 == set2) { 338 return true; 339 } 340 if (set1 == null || set2 == null || set1.size() != set2.size()) { 341 return false; 342 } 343 344 return set1.containsAll(set2); 345 } 346 347 /** 348 * Returns a new hash set that matches elements based on <code>==</code> not 349 * <code>equals()</code>. 350 * <p> 351 * <strong>This set will violate the detail of various Set contracts.</strong> 352 * As a general rule, don't compare this set to other sets. In particular, you can't 353 * use decorators like {@link ListOrderedSet} on it, which silently assume that these 354 * contracts are fulfilled. 355 * <p> 356 * <strong>Note that the returned set is not synchronized and is not thread-safe.</strong> 357 * If you wish to use this set from multiple threads concurrently, you must use 358 * appropriate synchronization. The simplest approach is to wrap this map 359 * using {@link java.util.Collections#synchronizedSet(Set)}. This class may throw 360 * exceptions when accessed by concurrent threads without synchronization. 361 * 362 * @param <E> the element type 363 * @return a new identity hash set 364 * @since 4.1 365 */ 366 public static <E> Set<E> newIdentityHashSet() { 367 return Collections.newSetFromMap(new IdentityHashMap<E, Boolean>()); 368 } 369 370 /** 371 * Returns a set that maintains the order of elements that are added 372 * backed by the given set. 373 * <p> 374 * If an element is added twice, the order is determined by the first add. 375 * The order is observed through the iterator or toArray. 376 * 377 * @param <E> the element type 378 * @param set the set to order, must not be null 379 * @return an ordered set backed by the given set 380 * @throws NullPointerException if the set is null 381 */ 382 public static <E> Set<E> orderedSet(final Set<E> set) { 383 return ListOrderedSet.listOrderedSet(set); 384 } 385 386 /** 387 * Returns a predicated (validating) navigable set backed by the given navigable set. 388 * <p> 389 * Only objects that pass the test in the given predicate can be added to the set. 390 * Trying to add an invalid object results in an IllegalArgumentException. 391 * It is important not to use the original set after invoking this method, 392 * as it is a backdoor for adding invalid objects. 393 * 394 * @param <E> the element type 395 * @param set the navigable set to predicate, must not be null 396 * @param predicate the predicate for the navigable set, must not be null 397 * @return a predicated navigable set backed by the given navigable set 398 * @throws NullPointerException if the set or predicate is null 399 * @since 4.1 400 */ 401 public static <E> SortedSet<E> predicatedNavigableSet(final NavigableSet<E> set, 402 final Predicate<? super E> predicate) { 403 return PredicatedNavigableSet.predicatedNavigableSet(set, predicate); 404 } 405 406 /** 407 * Returns a predicated (validating) set backed by the given set. 408 * <p> 409 * Only objects that pass the test in the given predicate can be added to the set. 410 * Trying to add an invalid object results in an IllegalArgumentException. 411 * It is important not to use the original set after invoking this method, 412 * as it is a backdoor for adding invalid objects. 413 * 414 * @param <E> the element type 415 * @param set the set to predicate, must not be null 416 * @param predicate the predicate for the set, must not be null 417 * @return a predicated set backed by the given set 418 * @throws NullPointerException if the set or predicate is null 419 */ 420 public static <E> Set<E> predicatedSet(final Set<E> set, final Predicate<? super E> predicate) { 421 return PredicatedSet.predicatedSet(set, predicate); 422 } 423 424 /** 425 * Returns a predicated (validating) sorted set backed by the given sorted set. 426 * <p> 427 * Only objects that pass the test in the given predicate can be added to the set. 428 * Trying to add an invalid object results in an IllegalArgumentException. 429 * It is important not to use the original set after invoking this method, 430 * as it is a backdoor for adding invalid objects. 431 * 432 * @param <E> the element type 433 * @param set the sorted set to predicate, must not be null 434 * @param predicate the predicate for the sorted set, must not be null 435 * @return a predicated sorted set backed by the given sorted set 436 * @throws NullPointerException if the set or predicate is null 437 */ 438 public static <E> SortedSet<E> predicatedSortedSet(final SortedSet<E> set, 439 final Predicate<? super E> predicate) { 440 return PredicatedSortedSet.predicatedSortedSet(set, predicate); 441 } 442 443 // Set 444 //----------------------------------------------------------------------- 445 /** 446 * Returns a synchronized set backed by the given set. 447 * <p> 448 * You must manually synchronize on the returned set's iterator to 449 * avoid non-deterministic behavior: 450 * 451 * <pre> 452 * Set s = SetUtils.synchronizedSet(mySet); 453 * synchronized (s) { 454 * Iterator i = s.iterator(); 455 * while (i.hasNext()) { 456 * process (i.next()); 457 * } 458 * } 459 * </pre> 460 * 461 * This method is just a wrapper for {@link Collections#synchronizedSet(Set)}. 462 * 463 * @param <E> the element type 464 * @param set the set to synchronize, must not be null 465 * @return a synchronized set backed by the given set 466 * @throws NullPointerException if the set is null 467 */ 468 public static <E> Set<E> synchronizedSet(final Set<E> set) { 469 return Collections.synchronizedSet(set); 470 } 471 472 // SortedSet 473 //----------------------------------------------------------------------- 474 /** 475 * Returns a synchronized sorted set backed by the given sorted set. 476 * <p> 477 * You must manually synchronize on the returned set's iterator to 478 * avoid non-deterministic behavior: 479 * 480 * <pre> 481 * Set s = SetUtils.synchronizedSortedSet(mySet); 482 * synchronized (s) { 483 * Iterator i = s.iterator(); 484 * while (i.hasNext()) { 485 * process (i.next()); 486 * } 487 * } 488 * </pre> 489 * 490 * This method is just a wrapper for {@link Collections#synchronizedSortedSet(SortedSet)}. 491 * 492 * @param <E> the element type 493 * @param set the sorted set to synchronize, must not be null 494 * @return a synchronized set backed by the given set 495 * @throws NullPointerException if the set is null 496 */ 497 public static <E> SortedSet<E> synchronizedSortedSet(final SortedSet<E> set) { 498 return Collections.synchronizedSortedSet(set); 499 } 500 501 /** 502 * Returns a transformed navigable set backed by the given navigable set. 503 * <p> 504 * Each object is passed through the transformer as it is added to the 505 * Set. It is important not to use the original set after invoking this 506 * method, as it is a backdoor for adding untransformed objects. 507 * <p> 508 * Existing entries in the specified set will not be transformed. 509 * If you want that behaviour, see {@link TransformedNavigableSet#transformedNavigableSet}. 510 * 511 * @param <E> the element type 512 * @param set the navigable set to transform, must not be null 513 * @param transformer the transformer for the set, must not be null 514 * @return a transformed set backed by the given set 515 * @throws NullPointerException if the set or transformer is null 516 * @since 4.1 517 */ 518 public static <E> SortedSet<E> transformedNavigableSet(final NavigableSet<E> set, 519 final Transformer<? super E, ? extends E> transformer) { 520 return TransformedNavigableSet.transformingNavigableSet(set, transformer); 521 } 522 523 /** 524 * Returns a transformed set backed by the given set. 525 * <p> 526 * Each object is passed through the transformer as it is added to the 527 * Set. It is important not to use the original set after invoking this 528 * method, as it is a backdoor for adding untransformed objects. 529 * <p> 530 * Existing entries in the specified set will not be transformed. 531 * If you want that behaviour, see {@link TransformedSet#transformedSet}. 532 * 533 * @param <E> the element type 534 * @param set the set to transform, must not be null 535 * @param transformer the transformer for the set, must not be null 536 * @return a transformed set backed by the given set 537 * @throws NullPointerException if the set or transformer is null 538 */ 539 public static <E> Set<E> transformedSet(final Set<E> set, 540 final Transformer<? super E, ? extends E> transformer) { 541 return TransformedSet.transformingSet(set, transformer); 542 } 543 544 /** 545 * Returns a transformed sorted set backed by the given set. 546 * <p> 547 * Each object is passed through the transformer as it is added to the 548 * Set. It is important not to use the original set after invoking this 549 * method, as it is a backdoor for adding untransformed objects. 550 * <p> 551 * Existing entries in the specified set will not be transformed. 552 * If you want that behaviour, see {@link TransformedSortedSet#transformedSortedSet}. 553 * 554 * @param <E> the element type 555 * @param set the set to transform, must not be null 556 * @param transformer the transformer for the set, must not be null 557 * @return a transformed set backed by the given set 558 * @throws NullPointerException if the set or transformer is null 559 */ 560 public static <E> SortedSet<E> transformedSortedSet(final SortedSet<E> set, 561 final Transformer<? super E, ? extends E> transformer) { 562 return TransformedSortedSet.transformingSortedSet(set, transformer); 563 } 564 565 // Set operations 566 //----------------------------------------------------------------------- 567 568 /** 569 * Returns a unmodifiable <b>view</b> of the union of the given {@link Set}s. 570 * <p> 571 * The returned view contains all elements of {@code a} and {@code b}. 572 * 573 * @param <E> the generic type that is able to represent the types contained 574 * in both input sets. 575 * @param a the first set, must not be null 576 * @param b the second set, must not be null 577 * @return a view of the union of the two set 578 * @throws NullPointerException if either input set is null 579 * @since 4.1 580 */ 581 public static <E> SetView<E> union(final Set<? extends E> a, final Set<? extends E> b) { 582 if (a == null || b == null) { 583 throw new NullPointerException("Sets must not be null."); 584 } 585 586 final SetView<E> bMinusA = difference(b, a); 587 588 return new SetView<E>() { 589 @Override 590 public boolean contains(final Object o) { 591 return a.contains(o) || b.contains(o); 592 } 593 594 @Override 595 public Iterator<E> createIterator() { 596 return IteratorUtils.chainedIterator(a.iterator(), bMinusA.iterator()); 597 } 598 599 @Override 600 public boolean isEmpty() { 601 return a.isEmpty() && b.isEmpty(); 602 } 603 604 @Override 605 public int size() { 606 return a.size() + bMinusA.size(); 607 } 608 }; 609 } 610 611 // NavigableSet 612 //----------------------------------------------------------------------- 613 /** 614 * Returns an unmodifiable navigable set backed by the given navigable set. 615 * <p> 616 * This method uses the implementation in the decorators subpackage. 617 * 618 * @param <E> the element type 619 * @param set the navigable set to make unmodifiable, must not be null 620 * @return an unmodifiable set backed by the given set 621 * @throws NullPointerException if the set is null 622 * @since 4.1 623 */ 624 public static <E> SortedSet<E> unmodifiableNavigableSet(final NavigableSet<E> set) { 625 return UnmodifiableNavigableSet.unmodifiableNavigableSet(set); 626 } 627 628 /** 629 * Create an unmodifiable set from the given items. If the passed var-args argument is {@code 630 * null}, then the method returns {@code null}. 631 * @param <E> the element type 632 * @param items the elements that make up the new set 633 * @return a set 634 * @since 4.3 635 */ 636 public static <E> Set<E> unmodifiableSet(final E... items) { 637 if (items == null) { 638 return null; 639 } 640 return UnmodifiableSet.unmodifiableSet(hashSet(items)); 641 } 642 643 /** 644 * Returns an unmodifiable set backed by the given set. 645 * <p> 646 * This method uses the implementation in the decorators subpackage. 647 * 648 * @param <E> the element type 649 * @param set the set to make unmodifiable, must not be null 650 * @return an unmodifiable set backed by the given set 651 * @throws NullPointerException if the set is null 652 */ 653 public static <E> Set<E> unmodifiableSet(final Set<? extends E> set) { 654 return UnmodifiableSet.unmodifiableSet(set); 655 } 656 657 /** 658 * Returns an unmodifiable sorted set backed by the given sorted set. 659 * <p> 660 * This method uses the implementation in the decorators subpackage. 661 * 662 * @param <E> the element type 663 * @param set the sorted set to make unmodifiable, must not be null 664 * @return an unmodifiable set backed by the given set 665 * @throws NullPointerException if the set is null 666 */ 667 public static <E> SortedSet<E> unmodifiableSortedSet(final SortedSet<E> set) { 668 return UnmodifiableSortedSet.unmodifiableSortedSet(set); 669 } 670 671 /** 672 * <code>SetUtils</code> should not normally be instantiated. 673 */ 674 private SetUtils() {} 675}