简单地:
public static class MyClass<T> {
// i don't want to keep an instance of T, if it is not necessary.
// and it is not nice, not neat.
// Or, let's say, the member are in the form of :
ArrayList<T> mArrayList = new ArrayList<T>();
// the problem of getting the generic type parameter is still present.
}
@Test
public final void test() {
MyClass<Integer> myObject = new MyClass<Integer>();
getParamType( myObject );
}
private static final <T> void getParamType(final MyClass<T> _myObject) {
System.out.println(_myObject.getClass().getTypeParameters()[0]); // T
System.out.println(((T) new Object()).getClass()); // class java.lang.Object
}
如何让代码打印class java.lang.Integer
?
我知道很多 stackoverflow 线程都在询问(并回答)这个问题。然而他们无法解决这个问题。
- 我不知道为什么有些人需要调用
getGenericSuperclass()
- 因为在这个简单的案例中不涉及继承。 - 而且我也无法将其投射到
ParameterizedType
。
.
System.out.println((ParameterizedType) _myObject.getClass());
// Compile Error: Cannot cast from Class<capture#11-of ? extends TreeTest2.MyClass> to ParameterizedType
System.out.println((ParameterizedType) _myObject.getClass().getGenericSuperclass());
// Runtime Exception: java.lang.ClassCastException
根据@Thomas 的指南,我找到了一种解决方法来获取class java.lang.Integer
.
首先,我们在测试代码中创建一个匿名(必须是匿名的)子类。MyClass<T>
(这很奇怪。为什么它只支持子类?)
@Test
public final void test() {
MyClass<Integer> myObject = new MyClass<Integer>() {}; // Anonymous sub-class
getParamType( myObject );
}
然后我们可以使用该getGenericSuperclass()
方法获取Type
然后将其转换为ParameterizedType
然后使用getActualTypeArguments()
:
private static final <T> void getParamType(final MyClass<T> _myObject) {
System.out.println( ((ParameterizedType) _myObject.getClass().getGenericSuperclass()).getActualTypeArguments()[0] );
}
它完美打印class java.lang.Integer
。
这不太好,因为测试代码应该模拟实际情况,用户很可能不会继续创建无意义的子类。
这种方法基于TypeReference 类的思想。但我真的不知道如何使用它。我试过了class MyClass<T> extends TypeReference<T>
。但我仍然必须创建子类MyClass<T>
才能拥有TypeReference.getType()
prints class java.lang.Integer
。
请提供帮助,并感谢您的任何意见,因为最好的方法还没有出现。
基于上述解决方法的另一个问题:为什么只有匿名子类有效?
public static class SubMyClass<T> extends MyClass<T>{}
@Test
public final void test() {
MyClass<Integer> myObject = new MyClass<Integer>() {}; // Anonymous sub-class
getParamType( myObject ); // class java.lang.Integer
MyClass<Integer> mySubObject = new SubMyClass<Integer>(); // named sub-class
getParamType( mySubObject ); // T
}
(MyClass
并且getParamType()
不变。)