Java:无法在枚举:初始化错误中使用EnumSet:Tech Research Talent Tree示例

use*_*573 12 java enums casting initialization

错误:

...
Caused by: java.lang.ExceptionInInitializerError
...
Caused by: java.lang.ClassCastException: 
class com.evopulse.ds2150.TechTrees$BuildingTechTree
not an enum
at java.util.EnumSet.noneOf(Unknown Source)
at java.util.EnumSet.of(Unknown Source)
at com.evopulse.ds2150.TechTrees$BuildingTechTree.<clinit>(TechTrees.java:38)
Run Code Online (Sandbox Code Playgroud)

这是我的枚举的片段

public enum BuildingTechTree {
//Name                      SoftName                    Requirements    
NONE                        ("NULL",                    null),
Run Code Online (Sandbox Code Playgroud)

- >下一行是它崩溃的地方

BARRACKS                    ("Barracks",                EnumSet.of(NONE),
WALLS_SANDBAGS              ("Sandbag wall",            EnumSet.of(NONE),

POWERPLANT                  ("Power plant",             EnumSet.of(BARRACKS)),
GUARDTOWER                  ("Guard Tower",             EnumSet.of(BARRACKS));
Run Code Online (Sandbox Code Playgroud)

用null替换EnumSet.of(NONE)和EnumSet.of(BARRACKS),让初始化工作,但由于缺少数据结构而破坏了我的代码......很明显,但是我做了它来测试我的其余代码不是不知何故原因.

删除EnumSet.of(NONE)并替换为NONE,并且相同的BARRACKS,并更改所有相关变量,构造函数和方法,这些都不起作用......(甚至无法使用contains.all ,因为是"不适用于我改变的变量"...)

我使用第二个实现扩展了这个例子:https: //gamedev.stackexchange.com/a/25652/48573

我也尝试通过逐字复制示例来回溯我的步骤.添加

private static Set<BuildingTechTree> techsKnown;

techsKnown = (BuildingTechTree.BIODOME);
test = TechTrees.researchTech(techsKnown);
Run Code Online (Sandbox Code Playgroud)

到另一个要调用初始化的类.并且不得不改变

public boolean researchTech(BuildingTechTree tech) {
Run Code Online (Sandbox Code Playgroud)

静止的

这导致了相同的"in en enum"错误.我没有任何代表,评论他的答案,指出初始化错误......

添加了当前答案的信息,因为两种解决方案都会导致相同的新错误:

public class TechTrees {
private static Set<BuildingTechTree> techsKnown;

public TechTrees() {
    techsKnown = EnumSet.of(BuildingTechTree.NONE);       //Using this
    techsKnown = EnumSet.noneOf(BuildingTechTree.class);  //Or this
}

public static boolean researchTech(BuildingTechTree tech) {
    if (techsKnown.containsAll(tech.requirements)) {      //Causes null pointer
        return true;                                      //exception @ techsKnown  
    }
    return false;
}
Run Code Online (Sandbox Code Playgroud)

Boa*_*ann 10

你的声明结构是如此聪明,这是一种耻辱,它不起作用.但EnumSet显然需要首先完全初始化枚举.它尝试从枚举中获取常量数组,以便除其他外,它知道其内部位集需要多少空间.

这是一个解决方法.它使用一个辅助方法,首先创建一个普通的set(HashSet),然后,在静态初始化块中,它迭代枚举常量并用EnumSets 替换所有的set .

public enum BuildingTechTree {
    // Named constants
    //Name                      SoftName                        Requirements
    NONE                        ("NULL",                        null),
    BARRACKS                    ("Barracks",                    setOf(NONE)),
    WALLS_SANDBAGS              ("Sandbag wall",                setOf(NONE)),
    POWERPLANT                  ("Power plant",                 setOf(BARRACKS)),
    GUARDTOWER                  ("Guard Tower",                 setOf(BARRACKS));

    private final String softName;
    private Set<BuildingTechTree> requirements;

    private BuildingTechTree(String softName, Set<BuildingTechTree> requirements) {
        this.softName = softName;
        this.requirements = requirements;
    }

    private static Set<BuildingTechTree> setOf(BuildingTechTree... values) {
        return new HashSet<>(Arrays.asList(values));
    }

    static {
        for (BuildingTechTree v : values()) {
            if (v.requirements == null) {
                v.requirements = EnumSet.noneOf(BuildingTechTree.class);
            } else {
                v.requirements = EnumSet.copyOf(v.requirements);
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)


JB *_*zet 6

你有鸡和蛋的问题.你可以将你的枚举重构为这样的东西:

public enum BuildingTechTree {

    NONE("NULL"),
    BARRACKS("Barracks"),
    WALLS_SANDBAGS("Sandbag wall"),
    POWERPLANT("Power plant"),
    GUARDTOWER("Guard Tower");

    static {
        NONE.trees = EnumSet.noneOf(BuildingTechTree.class);
        BARRACKS.trees = EnumSet.of(NONE);
        WALLS_SANDBAGS.trees = EnumSet.of(NONE);
        POWERPLANT.trees = EnumSet.of(BARRACKS);
        GUARDTOWER.trees = EnumSet.of(BARRACKS);
    }

    private String name;
    private Set<BuildingTechTree> trees;

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

    public String getName() {
        return name;
    }

    public Set<BuildingTechTree> getTrees() {
        return Collections.unmodifiableSet(trees);
    }
}
Run Code Online (Sandbox Code Playgroud)

编辑:

关于你的第二个问题:你是从静态方法访问一个静态变量.但是当调用类的构造函数时,这个变量被初始化(这是一个巨大的设计问题).不要使用非最终的静态字段.并且不要从实例方法或构造函数初始化静态字段.这没有意义.您没有设置构建汽车时所有汽车应具有的颜色.静态初始化静态字段:

public class TechTrees {
    private static final Set<BuildingTechTree> TECHS_KNOWN =
        EnumSet.of(BuildingTechTree.NONE);

    public static boolean researchTech(BuildingTechTree tech) {
        return TECHS_KNOWN.containsAll(tech.requirements));
    }
}
Run Code Online (Sandbox Code Playgroud)