EnumSet from array, shortest variant?

13,899

Solution 1

This is two lines, but slightly less complex:

EnumSet<Options> temp = EnumSet.noneOf(Options.class); // make an empty enumset
temp.addAll(Arrays.asList(options)); // add varargs to it

I don't consider this worse than any other kind of variable declaration for a class which doesn't have the constructor you want:

    SomeClass variable = new SomeClass(); // make an empty object
    variable.addStuff(stuff); // add stuff to it

Solution 2

Just an alternative. Same amount of code, except no need converting to list, using EnumSet.of() :

EnumSet<Options> temp = options.length > 0 ? 
                        EnumSet.of(options[0], options) : 
                        EnumSet.noneOf(Options.class);

No worries that first element is repeated(it won't be duplicated in a set anyway), no performance gain or penalties as well.

Solution 3

Guava has factory methods for these kinds of situations: It still needs the call to Arrays.asList but at least it's readable.

import com.google.common.collect.Sets;

EnumSet<Options> temp = Sets.newEnumSet(Arrays.asList(options), Options.class);

Solution 4

You can also do this with Java8 streams and your own CollectionFactory:

EnumSet<Options> temp = Arrays.stream(options)
        .collect(Collectors.toCollection(() -> EnumSet.noneOf(Options.class)));

Solution 5

The simplest solution is the obvious one: If you have access to the method's signature you are using then just mimic the signature of EnumSet.of(T first, T ... more) as

void myMethod(Options first, Options ... rest) {
    EnumSet<Options> temp = EnumSet.of(first, rest);
}

As the ellipsis is already the last element of your method's parameters list (because it has to be) this won't change any calling code and you don't need to jump through loops.

Share:
13,899
qqilihq
Author by

qqilihq

Co-founder and Cee-something-Ohh at LineUpr: Create an app for virtual, hybrid and physical events in thirty minutes. Developer of the Selenium KNIME Nodes: Automate your browser with a graphical workflow without writing a line of code. One of the developers behind NodePit: Your search engine for KNIME nodes and workflows.

Updated on June 06, 2022

Comments

  • qqilihq
    qqilihq almost 2 years

    I need an EnumSet from an array (which is given through a varargs method parameter). First, I was surprised that there is no varargs constructor method in EnumSet (there is EnumSet#of(E first, E... rest)). As a workaround, I used the following variant:

    EnumSet<Options> temp = EnumSet.copyOf(Arrays.asList(options));
    

    However, this triggers a java.lang.IllegalArgumentException: Collection is empty. So, now I ended up with the following, which looks somewhat ridiculous:

    EnumSet<Options> temp = options.length > 0 ? 
                            EnumSet.copyOf(Arrays.asList(options)) : 
                            EnumSet.noneOf(Options.class);
    

    If course this could be moved to some utility method, but still, I'm asking myself if there is a simpler way using existing methods?

  • Radon Rosborough
    Radon Rosborough over 9 years
    This doesn't address the case in which the array is empty.
  • gouessej
    gouessej about 6 years
    Why adding a third party library just to make exactly what could already been done with the Java standard API in 2014? dkarp's answer is better to me because it doesn't use a library, it doesn't create an intermediary list and I find it readable even though the single line of code is a bit longer. Moreover, Guava has a serious lack of modularity which might refrain some developers from using it.
  • Basil Bourque
    Basil Bourque about 6 years
    @gouessej While the points in your comment are valid, this Answer is still helpful and informative. Seems especially on point as the Question specifically mentions surprise at the lack of a no varargs constructor – this Answer with Guava gives the same effect, a single line with similar syntax and readability. I upvoted all of the Answers on this page as they were all clever or informative.
  • gouessej
    gouessej about 6 years
    @BasilBourque I'll go on downvoting answers that use third party dependencies in any context in which it isn't required (2014, Java 1.8 available) and in which it's not explicitly asked by the original poster (no "guava" tag, no mention of Guava in the question).
  • Johan Tidén
    Johan Tidén about 6 years
    To be fair. There is nothing in the question stating that they have java 1.8. Also, library answers might be useful for readers that already happen to use that lib. I would assume that readers are mature enough to make their own decision whether or not to include a new lib.
  • Mark Jeronimus
    Mark Jeronimus over 5 years
    Collections.addAll(temp, options); is cleaner than temp.addAll(Arrays.asList(options));
  • OldCurmudgeon
    OldCurmudgeon about 4 years
    Nice insight EnumSet.of(options[0], options)! Works with generics too :)
  • Splash
    Splash over 3 years
    myEnumSet = EnumSet.copyOf(Arrays.asList(myArray)); // myArray is not empty