使用 java 泛型迭代枚举值

2022-08-31 10:43:43

我试图找到一种方法来在使用泛型时循环访问枚举的值。不知道如何做到这一点,或者是否可能。

下面的代码阐释了我想要执行的操作。请注意,代码 T.values() 在下面的代码中无效。

public class Filter<T> {
    private List<T> availableOptions = new ArrayList<T>();
    private T selectedOption;

    public Filter(T selectedOption) {
        this.selectedOption = selectedOption;
        for (T option : T.values()) {  // INVALID CODE
            availableOptions.add(option);
        }
    }
}

以下是我如何实例化 Filter 对象:

Filter<TimePeriod> filter = new Filter<TimePeriod>(TimePeriod.ALL);

枚举定义如下:

public enum TimePeriod {
    ALL("All"), 
    FUTURE("Future"), 
    NEXT7DAYS("Next 7 Days"), 
    NEXT14DAYS("Next 14 Days"), 
    NEXT30DAYS("Next 30 Days"), 
    PAST("Past"),
    LAST7DAYS("Last 7 Days"), 
    LAST14DAYS("Last 14 Days"),
    LAST30DAYS("Last 30 Days"); 

    private final String name;

    private TimePeriod(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return name;
    }
}

我意识到将枚举的值复制到列表中可能没有意义,但是我正在使用一个需要值列表作为输入并且不能与枚举一起使用的库。


编辑 2/5/2010:

大多数建议的答案都非常相似,并建议做这样的事情:

class Filter<T extends Enum<T>> {
    private List<T> availableOptions = new ArrayList<T>();
    private T selectedOption;

    public Filter(T selectedOption) {
        Class<T> clazz = (Class<T>) selectedOption.getClass();
        for (T option : clazz.getEnumConstants()) {
            availableOptions.add(option);
        }
    }
}

如果我能确定 selectedOption 具有非空值,这将非常有效。不幸的是,在我的用例中,这个值通常是空的,因为还有一个公共的Filter()no-arg构造函数。这意味着我不能在没有获得NPE的情况下做一个selectedOption.getClass()。此筛选器类管理选择了哪些选项的可用选项的列表。如果未选择任何内容,则所选选项为空。

我唯一能想到的解决这个问题就是在构造函数中实际传递一个类。所以像这样:

class Filter<T extends Enum<T>> {
    private List<T> availableOptions = new ArrayList<T>();
    private T selectedOption;

    public Filter(Class<T> clazz) {
        this(clazz,null);
    }

    public Filter(Class<T> clazz, T selectedOption) {
        this.selectedOption = selectedOption;
        for (T option : clazz.getEnumConstants()) {
            availableOptions.add(option);
        }
    }
}

任何想法如何做到这一点,而不需要在构造函数中额外的Class参数?


答案 1

这确实是一个难题。你需要做的一件事是告诉java你正在使用枚举。这是通过声明您为泛型扩展 Enum 类来实现的。但是,此类没有 values() 函数。因此,您必须参加可以获得其值的类。

以下示例应可帮助您解决问题:

public <T extends Enum<T>> void enumValues(Class<T> enumType) {
        for (T c : enumType.getEnumConstants()) {
             System.out.println(c.name());
        }
}

答案 2

另一种选择是使用EnumSet:

class PrintEnumConsants {

    static <E extends Enum <E>> void foo(Class<E> elemType) {
        for (E e : java.util.EnumSet.allOf(elemType)) {
            System.out.println(e);
        }
    }

    enum Color{RED,YELLOW,BLUE};
    public static void main(String[] args) {
        foo(Color.class);
    } 

}