在运行时检查类是否具有使用泛型的特定构造函数

时间:2013-04-17 15:14:24

标签: java generics reflection constructor introspection

Hello all :)我正在尝试在类中选择正确的构造函数。这是代码:

Constructor[] constructors = targetClass.getConstructors();
Constructor goodConstructor = null;
for (Constructor constructor : constructors) {
    Class[] parameterTypes = constructor.getParameterTypes();
    if (parameterTypes.length = 1 && parameterTypes[0].equals(Map.class)) {//here
        goodConstructor = constructor;
    }
}

我想从Map.class切换到Map<String, String>.class。我依稀记得泛型仅用于编译时,所以这就是编译器抱怨的原因。如何在运行时检查该类是否具有正确的构造函数?

祝你好运

2 个答案:

答案 0 :(得分:8)

您想要使用getGenericParameterTypes()代替:

public class FindConstructor {

    public static void main(String[] args) throws IOException {
        for (Constructor<?> constructor : MyClass.class.getConstructors()) {
            Type[] parameterTypes = constructor.getGenericParameterTypes();
            if (parameterTypes.length == 1 && parameterTypes[0] instanceof ParameterizedType) {
                ParameterizedType parameterizedArg = (ParameterizedType) parameterTypes[0];
                if (parameterizedArg.getRawType() != Map.class) {
                    continue;
                }

                if (parameterizedArg.getActualTypeArguments()[0] != String.class) {
                    continue;
                }

                if (parameterizedArg.getActualTypeArguments()[1] != String.class) {
                    continue;
                }
            }
            System.out.println("found constructor " + constructor);
        }
    }
}

class MyClass {
    public MyClass(Map<String, String> map) {
    }
}

现在,如果您将MyClass()更改为Map<String, Integer>,则它将不再匹配。

使用Guava的TypeToken会变得更容易,它使用匿名类来创建我们可以比较的参数化Type

Type mapStringString = new TypeToken<Map<String, String>>(){}.getType();
for (Constructor<?> constructor : MyClass.class.getConstructors()) {
    Type[] parameterTypes = constructor.getGenericParameterTypes();
    if (parameterTypes.length == 1 && parameterTypes[0].equals(mapStringString)) {
        System.out.println("found constructor " + constructor);
    }
}

答案 1 :(得分:0)

类在java中不起作用。类不能在运行时参数化;只有实例。只有一个Map.class,而不是(你似乎已经假设)一个单独的类用于Map的每个泛型实现。以下代码可能有助于澄清这一点:

    Map<String,String> m1 = new HashMap<>();
    Map<Object,Long> m2 = new HashMap<>();

    System.out.println(m1.getClass() == m2.getClass());//prints "true"