我正在寻找一种方法来确定对象x
是否是一个实例
通用类型。例如List<String>
。
受Super-Type-Token习语的启发,我可以在运行时使用以下代码检索java.lang.reflect.Type
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.List;
public class Test {
public static void main(String[] args) {
TypeReference<List<String>> ref = new TypeReference<List<String>>() {};
System.out.println(ref.getType());
System.out.println(ref.getType().getClass());
}
abstract static class TypeReference<T> {
private final Type type;
protected TypeReference() {
ParameterizedType superclass = (ParameterizedType) getClass().getGenericSuperclass();
type = superclass.getActualTypeArguments()[0];
}
public Type getType() {
return type;
}
}
}
输出
java.util.List<java.lang.String>
class sun.reflect.generics.reflectiveObjects.ParameterizedTypeImpl
如何将此信息用于动态投射或类型检查?
背景是我目前正在编写异构的类型安全容器库,并希望添加对泛型类型的支持https://bitbucket.org/mbeyene/jam
答案 0 :(得分:1)
您必须注意到您正在使用参数化类型作为TypeReference<T>
实例的类型参数。因此,您需要将其转发到调用者端的ParameterizedType
,并从中获取原始类型。
然后您可以将该rawtype转换为Class<?>
类型,并使用Class#isInstance()
方法:
public static void main(String[] args) {
TypeReference<List<String>> ref = new TypeReference<List<String>>() {};
List<String> list = new ArrayList<String>();
Type rawType = ((ParameterizedType)ref.getType()).getRawType();
boolean listIsInstanceOfRawType = ((Class<?>)(rawType)).isInstance(list));
System.out.println(listIsInstanceOfRawType); // true
}
请注意,您无法针对参数化类型List<String>
或List<Integer>
检查instanceof,因为这没有意义。它们都只是List
在运行时。我的意思是:
List<String> list = new ArrayList<String>();
System.out.println(list instanceof List<String>); // Won't compile
System.out.println(list instanceof List); // You've to do this.
// The reason is, this is true
System.out.println(new ArrayList<String>().getClass() == new ArrayList<Integer>().getClass()); // true