使用Class对象

时间:2017-07-26 23:02:40

标签: java generics reflection collections

如何实例化Java泛型对象,该对象只接受ClassClass<?>对象的类型参数?

例如: 通常,可以使用以下语法实例化ArrayListInteger个对象:

ArrayList<Integer> foo = new ArrayList<Integer>();

但是,如果有Class<?> Integer.class对象,那么如何创建类似的ArrayList呢?例如,我将如何做这样的事情(语法不正确):

ArrayList<Integer.class> foo = new ArrayList<Integer.class>();

我需要这个用于Java的非常不寻常的事情(创建一个开源工具,用于可视化用户提供的数据结构实例/他们编写的泛型类)。以下是我将如何使用此代码的示例,该代码说明了我将获得的信息:

import java.util.ArrayList;
import java.util.List;

public class ArrayListFromClass {
    // obviously this code does not work
    public static void main(String[] args) {
        Object givenObject = new Integer(4);
        // I would not know it is Integer.class, a Class<?> object would be supplied by the user/ as a generic
        Class<?> cls = givenObject.getClass();
        List<cls> bar = new ArrayList<cls>();

        // Where args[0] is "Integer.class"
        List<args[0]> foo = new ArrayList<args[0]>();

        // then I would be able to add to foo or bar with one or both of these techniques:
        // printing givenObject.getClass() gives you java.lang.Integer, don't worry about the casting not working.
        bar.add(cls.cast(givenObject));
        Integer y = 6;
        bar.add(y);
    }
}

2 个答案:

答案 0 :(得分:1)

Java是静态类型的。你想要像这样实例化一个ArrayList的原因:

ArrayList<Integer> foo = new ArrayList<>();

是让你的IDE /编译器知道每当Integer以外的东西放入列表时,显示错误。除此之外,内部将忽略此初始化,实际上甚至是erase the type

因此,当您在运行时获取Class<?>对象时,您只知道代码实际运行时类的类型。因此,IDE /编译器无法在运行时警告您代码中是否存在错误。

所以List<Object>对你的情况会很好。如果你想要类型安全,你必须自己检查:

    String sampleString = "String";
    Class<?> clazz = sampleString.getClass();
    ...
    if (clazz.isInstance(sampleString)) {
        list.add(sampleString);
    }
当您实际拥有.isInstance(Object obj)时,

instanceof相当于Class<?>

答案 1 :(得分:0)

您不必担心通用参数。 事实证明,通用参数实际上是在运行时擦除的(参见type erasure)。

所以你只需要做List<Object>

这似乎是一个问题,但它很少重要。您只需要跟踪您输入的类型,并确保正确删除和投射。大多数情况下,当您使用反射时,Object的具体类型 确实非常重要,因为您调用方法并间接访问字段。