如何使用反射检查对象的泛型类型



我需要检查类的字段是否是List<String>。我尝试使用以下代码执行此操作:

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

这似乎是一个错误的代码,因为即使在实际上是List<String>的字段上formField.getGenericType().equals((Class<List<String>>)(Class<?>)List.class)也会返回false

另一种方法是首先测试字段是否与formField.getType().equals(List.class) List,然后检查它是否是泛型的formField.getGenericType() instanceof ParameterizedType,最后检查类型参数的类。

但我认为有一种更短的方法来实现这种检查。有吗?

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

你可以像这样使用Guava的新TypeToken类(它将包含在12.0版本中......它的候选版本目前在Maven中可用):

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>字段,前提是您正在处理的类是在其类型中指定T是什么的子类。举个例子:

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的更多信息。

最新更新