如何在运行时获取generics类型?
这是我的代码:ExecutorImp扩展了AbstractExecutor,它提取了其实现者的相同执行逻辑(ExecutorImp就是一个例子),当调用ExecutorImp的execute()方法时,它将调用其超类型中的方法,但是超类型(AbstractExcutor) )应该知道另一个绑定到实现者的类(在示例中,它是User类):
import java.lang.reflect.InvocationTargetException; import java.util.ArrayList; abstract class AbstractExecutor { public void execute() throws Exception { ArrayList list = new ArrayList(); // here I want to get the real type of 'E' Class cl = this.getClass().getTypeParameters()[0].getGenericDeclaration().getClass(); Object o = cl.getConstructor(String.class).newInstance("Gate"); list.add((E) o); System.out.println(format(list)); } public abstract String format(ArrayList list); public abstract String getType(); } public class ExectorImp extends AbstractExecutor { @Override public String getType() { return "user"; } @Override public String format(ArrayList list) { StringBuffer sb = new StringBuffer(); for (User u : list) { sb.append(u.toString() + " "); } return sb.toString(); } public static void main(String[] args) throws Exception { new ExectorImp().execute(); } } class User { String name; public User(String name) { this.name = name; } }
那么,我的代码有什么问题?
这里有一些混乱。 由于类型擦除,您无法从运行时参数化类型获取类型信息,如:
Class cls = E.getClass(); // Error. E e = new E(); // Error.
但是,您可以通过ParameterizedType#getActualTypeArguments()
从类,字段和方法声明中获取编译时参数化类型信息。
abstract class AbstractExecutor { public void execute() throws Exception { List list = new ArrayList (); Class cls = (Class ) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0]; E e = cls.getConstructor(String.class).newInstance("Gate"); list.add(e); System.out.println(format(list)); } // ... }
更新 :关于是否建议这样做,尽管这会起作用,但只要发生类声明中的微小更改,这就会对运行时问题敏感。 您作为开发人员应该正确记录它。 作为一种完全不同的替代方案,您可以利用多态性。
abstract class AbstractExecutor { public void execute() throws Exception { List list = new ArrayList (); E e = create("Gate"); list.add(e); System.out.println(format(list)); } public abstract E create(String name); // ... }
并相应地实现UserExecutor
。
class UserExecutor extends AbstractExecutor { @Override public User create(String name) { return new User(name); } // ... }
我认为你应该使用getActualTypeParameters
; 因为getTypeParameters
没有引用当前实例化中代替E
,而是引用E
本身(描述它是如何限制的等等)。
为了获得ParameterizedType
您应该首先使用getGenericSuperclass
。
更新 :但上述仅在当前对象派生自generics参数实例化的generics类时才有效,如:
class StringList extends ArrayList { public Type whatsMyGenericType() { return ((ParameterizedType)getGenericSuperClass()).getActualTypeParameters()[0]; } }
应该返回String.class
。
我不认为你可以在运行时获得generics类型。 generics类型是在编译时应用的限制。 我记得在运行时,generics集合和没有generics类型的集合之间没有区别。
解决问题的常用方法是稍微更改代码。 在接受Class
参数的基类上定义构造函数。 将此参数分配给内部字段。
在子类上定义没有参数的构造函数,并从那里调用super(User.class)
。
通过这种方式,您将了解参数类,而不会为子类的客户端造成太多负担。