如何使用反射来查看对象的泛型类型?

4

我需要检查一个类的字段是否是 List<String>。我尝试使用以下代码:

for (Field formField : formClass.getDeclaredFields()) {    
  ...
  if (formField.getGenericType().equals((Class<List<String>>)(Class<?>)List.class)) {
    ...
  }
}

似乎这段代码有问题,因为 formField.getGenericType().equals((Class<List<String>>)(Class<?>)List.class) 即使在实际上是 List<String> 的字段上也会返回 false
另一种方法是先测试一个字段是否为 List,使用 formField.getType().equals(List.class),然后检查它是否是泛型的,使用 formField.getGenericType() instanceof ParameterizedType,最后检查类型参数的类。
但我认为有一种更简短的方式来实现这个检查。有吗?
2个回答

4
import java.lang.reflect.Field;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.List;

public class Main {
    private List<String> strings;
    private List<ReflectionUtils> list;
    public static void main(String[] args) {
        List<Class> genericTypes = ReflectionUtils.getGenericType(Main.class);
        for(Class genericType : genericTypes) {
            System.out.println(genericType.getName());
        }
    }
}

class ReflectionUtils {
    public static List<Class> getGenericType(Class clazz) {
        List<Class> genericTypes = new ArrayList<Class>();
        Field[] fields = clazz.getDeclaredFields();
        for(Field field : fields) {
            Class c = getGenericType(field);
            if(c != null) {
                genericTypes.add(c);
            }
        }
        return genericTypes;
    }

    public static Class getGenericType(Field field) {
        Type genericType = field.getGenericType();
        if(genericType instanceof ParameterizedType) {
            ParameterizedType parameterizedType = (ParameterizedType) genericType;
            return (Class) parameterizedType.getActualTypeArguments()[0];
        }
        return null;
    }
}

例子的输出结果:

java.lang.String
ReflectionUtils

3
你可以像这样使用TypeToken类(将在12.0版本中包含...它的发布候选版目前在Maven中可用)来使用Guava的新功能:
for (Field formField : formClass.getDeclaredFields()) {    
  ...
  TypeToken<?> type = TypeToken.of(formField.getGenericType());
  if (type.equals(new TypeToken<List<String>>(){}) {
    ...
  }
}

请注意,这仅适用于类字段实际声明为List<String>,而不是List<T>ArrayList<String>。如果您希望它匹配ArrayList<String>或任何其他实现List<String>的类型,则可以执行以下检查:
if (new TypeToken<List<String>>(){}.isAssignableFrom(type)) {

实际上,如果你处理的类是指定其类型的子类,则可以找出声明为List<T>List<String>字段是否在声明类中。看这个例子:

class Foo<T> {
  private List<T> list;
  ...
}

class StringFoo extends Foo<String> { // T is specified in here
  ...
}

TypeToken<StringFoo> stringFooType = TypeToken.of(StringFoo.class);
// iterate through superclasses of StringFoo
for (TypeToken<?> superclass : stringFooType.getTypes().classes()) {
  for (Field field : superclass.getRawType().getDeclaredFields()) {
    // actually resolves T to String for the List<T> field using info from
    // StringFoo's type
    TypeToken<?> type = stringFooType.resolveType(field.getGenericType());
    if (new TypeToken<List<String>>(){}.isAssignableFrom(type)) {
      System.out.println("List<String> found!");
    }
  }
}

关于TypeToken的更多信息,请查看这里


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接