我需要检查类的字段是否是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
的更多信息。