Reputation: 11607
I have an immutable set (cast as a Set<Integer>
) that potentially contains many elements. I need a Collection that contains the elements from that set plus one additional element. I have kludgy code in place to copy the set, then append the element, but I'm looking for The Right Way that keeps things as efficient as possible.
I have Guava available, though I do not require its use.
Upvotes: 46
Views: 36345
Reputation: 42223
Using Java 8 you can also use streams for that effect
Stream.concat(oldSet.stream(),
Stream.of(singleElement))
.collect(Collectors.toSet())
By the way since JDK 10, the Collectors
also allow to accumulate to immutable types (the same as the ones created by the static factories like Set.of()
) :
Stream.concat(oldSet.stream(),
Stream.of(singleElement))
.collect(Collectors.toUnmodifiableSet())
Upvotes: 23
Reputation: 120516
When you want better performance than a full copy, and you have an ordering over elements, you can use an effectively immutable wrapper around a B+ tree to get good incremental set performance.
Adding an item to a B+ tree requires O(log(n)) time and incremental allocation, not O(n) as you get with ImmutableSet.builder().addAll(...).add(...).build()
. This means that building a set from n incremental adds is O(n*log(n)), not O(sqr(n)).
This answer has a pointer to a jdbm library so it might be worth looking at jdbm:jdbm
.
Upvotes: 1
Reputation: 2121
You might consider Sets.union(). Construction would be faster, but use slower.
public static <T> Set<T> setWith(Set<T> old, T item) {
return Sets.union(old, Collections.singleton(item);
}
(com.google.common.collect.Sets & java.util.Collections)
Upvotes: 8
Reputation: 95318
Not sure about performance, but you can use Guava's ImmutableSet.Builder
:
import com.google.common.collect.ImmutableSet
// ...
Set<Integer> newSet = new ImmutableSet.Builder<Integer>()
.addAll(oldSet)
.add(3)
.build();
Of course you can also write yourself a helper method for that:
public static <T> Set<T> setWith(Set<T> old, T item) {
return new ImmutableSet.Builder<T>().addAll(old).add(item).build();
}
// ...
Set<Integer> newSet = setWith(oldSet, 3);
Upvotes: 45
Reputation: 533530
You have three options.
Sometimes a BitSet
is a better choice than Set<Integer>
depending on the distribution of your values.
Upvotes: 4
Reputation: 308813
I'm experiencing cognitive dissonance when I read "immutable" and "add to" in the same sentence. You can add a new element to the end of a mutable copy of the immutable values, but you can't modify the immutable set. I don't know of anything elegant.
Upvotes: -3
Reputation: 120198
If the Set is immutable, I don't see any way to do it other than copy the Set, and then add your new element. Remember, copying a set is as easy as passing the base set to the constructor function when creating the new set.
Upvotes: 2