如何实例化Java泛型对象,该对象只接受Class
或Class<?>
对象的类型参数?
例如:
通常,可以使用以下语法实例化ArrayList
个Integer
个对象:
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);
}
}
答案 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
的具体类型 确实非常重要,因为您调用方法并间接访问字段。