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)即使false在实际上是List<String>.

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

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

4

2 回答 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
于 2012-04-26T17:34:30.197 回答
3

您可以像这样使用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<String>字段。举个例子:List<T>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可用here

于 2012-04-26T19:39:43.480 回答