编程语言
首页 > 编程语言> > java – 枚举,类,反射和通用铸造

java – 枚举,类,反射和通用铸造

作者:互联网

考虑一下我有一个接口com.mycompany.SomeInterface,一个枚举com.mycompany.SomeEnum实现了SomeInterface.我希望在运行时使用Reflection API获取此类的所有枚举常量 – 作为SomeInterface的实例.

目前,我的代码(在类EnumConstantGetter中)是这样的:

Class<?> clazz = EnumConstantGetter.class.getClassLoader().
  loadClass("com.mycompany.SomeEnum");

if (!(SomeInterface.class.isAssignableFrom(clazz)) {
    throw new Exception("the class doesn't implement SomeInterface");
}

if (!(clazz.isEnum()) {
    throw new Exception("not an enum");
} 

Class<? extends Enum<? extends SomeInterface>> castClass =
  (Class<? extends Enum<? extends SomeInterface>>) clazz; // cast #1
ArrayList<SomeInterface> vals = new ArrayList<SomeInterface>();
for (Enum<? extends SomeInterface> enumConstant :
  castClass.getEnumConstants()) {
    vals.add((SomeInterface) enumConstant); // cast #2
}

上面的代码似乎有效,但我在创建castClass时收到编译器警告.

那么我的问题是:代码中注明的两个强制转换(类上的强制转换和常量上的强制转换)是否必须基于我的检查有效?

换句话说,是Enum的每个成员<?延伸T>保证实施T? 如果答案是“是的,演员阵容是安全的”,那么为什么编译器会给我这个警告呢? 如果没有,为什么不呢?或者,在什么情况下例程会失败? 编辑:由于上面的代码显然令人困惑,这里是我对应该发生的事情的解释:
>在com.mycompany包中加载名为SomeEnum的类,并将其Class对象存储在变量中.
>确保引用的类实现SomeInterface接口.
>确保引用的类是枚举.
>我们知道它是实现SomeInterface的枚举,将其强制转换为Class<?延伸Enum<?扩展SomeInterface>> – 演员阵容#1
>遍历所有枚举常量.
>对于每个常量,将其强制转换为SomeInterface – 强制转换#2 – 并将其添加到常量列表中.

谢谢!

解决方法:

您可以执行以下操作:

Class<?> clazz = ...;
Class<? extends SomeInterface> someInterfaceClass;
try {
    someInterfaceClass = clazz.asSubclass(SomeInterface.class);
}
catch (ClassCastException cce) {
    throw new IllegalStateException("Specified type doesn't implement SomeInterface", cce);
}
SomeInterface[] enumConstants = someInterfaceClass.getEnumConstants();
if (enumConstants == null) {
    throw new IllegalStateException("Specified type is not an enum.");
}

//use constants

这避免了警告,因为检查了asSubclass,如果Class对象“不表示枚举类型”,则getEnumConstants返回null.

标签:java,reflection,enums,casting,type-safety
来源: https://codeday.me/bug/20190901/1784081.html