使用Class对象实例化Java类型参数/ generic

参见英文答案 >
How to set the generic type of an ArrayList at runtime in java?                                    4个

如何实例化Java泛型对象,该对象仅接受类或参数给出的类型参数<?>宾语?

例如:
通常,可以使用以下语法实例化Integer对象的ArrayList:

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);
    }
}

最佳答案 Java是静态类型的.您希望实例化ArrayList的原因如下:

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

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

所以,当你得到你的班级<?>对象在运行时,您只知道代码实际运行时类的类型.因此,IDE /编译器无法在运行时警告您代码中是否存在错误.

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

    String sampleString = "String";
    Class<?> clazz = sampleString.getClass();
    ...
    if (clazz.isInstance(sampleString)) {
        list.add(sampleString);
    }

当你实际拥有一个Class<?>时,.isInstance(Object obj)相当于instanceof.在眼前.

点赞