java - 如何在 Java 中创建一个泛型数组?

标签 java arrays generics reflection instantiation

由于Java泛型的实现,你不能有这样的代码:

public class GenSet<E> {
    private E a[];

    public GenSet() {
        a = new E[INITIAL_ARRAY_LENGTH]; // error: generic array creation
    }
}

如何在保持类型安全的同时实现这一点?

我在 Java 论坛上看到了这样的解决方案:

import java.lang.reflect.Array;

class Stack<T> {
    public Stack(Class<T> clazz, int capacity) {
        array = (T[])Array.newInstance(clazz, capacity);
    }

    private final T[] array;
}

但我真的不明白发生了什么。

最佳答案

我必须问一个问题作为返回:你的 GenSet “检查”还是“未检查”? 这是什么意思?

  • 选中:强类型GenSet明确知道它包含什么类型的对象(即,它的构造函数是用 Class<E> 参数显式调用的,并且方法在传递非 E 类型的参数时会抛出异常。参见 Collections.checkedCollection

    -> 在这种情况下,你应该写:

    public class GenSet<E> {
    
        private E[] a;
    
        public GenSet(Class<E> c, int s) {
            // Use Array native method to create array
            // of a type only known at run time
            @SuppressWarnings("unchecked")
            final E[] a = (E[]) Array.newInstance(c, s);
            this.a = a;
        }
    
        E get(int i) {
            return a[i];
        }
    }
    
  • 未选中:弱输入。实际上没有对作为参数传递的任何对象进行类型检查。

    -> 在这种情况下,你应该写

    public class GenSet<E> {
    
        private Object[] a;
    
        public GenSet(int s) {
            a = new Object[s];
        }
    
        E get(int i) {
            @SuppressWarnings("unchecked")
            final E e = (E) a[i];
            return e;
        }
    }
    

    注意数组的组件类型应该是erasure类型参数:

    public class GenSet<E extends Foo> { // E has an upper bound of Foo
    
        private Foo[] a; // E erases to Foo, so use Foo[]
    
        public GenSet(int s) {
            a = new Foo[s];
        }
    
        ...
    }
    

所有这一切都源于 Java 中一个已知且有意为之的弱点:它是使用删除实现的,因此“泛型”类不知道它们在运行时使用什么类型参数创建,因此不能除非实现了某种显式机制(类型检查),否则提供类型安全。

关于java - 如何在 Java 中创建一个泛型数组?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/529085/

相关文章:

c# - 我可以在 C# 泛型约束中指定 'supertype' 关系吗?

java - System.console() 空指针异常

java - 在Java中递归调用方法时如何修改final变量

java - 为什么我在 Android Studio 中不断收到 java.lang.nullpointerException?

javascript - 在 Javascript 中查找引号内的字符串

java - 通过泛型对象类型访问类的方法

java - Spring jpa 异常 - 发现对集合的共享引用

arrays - 如何以最有效的方式在大小为 100 的整数数组中找到 0,其中 99 个元素为 1,只有一个元素为 0

java - 交换方法java的问题

c# - C# : operator '<' cannot be applied to operands of type 'T' and 'T' 中的比较