Java 必知必会 第 41 篇

如何创建一个java泛型数组

问题:

由于Java泛型的实现机制,你不能这样写代码:
pulic class GenSet{
private E a[];
public GetSet(){
a=new E[INITIAL_ARRAY_LENGTH]; //error:generic array creation
}
}
在保证类型安全的情况下,我该如何实现创建一个Java泛型数组?
我在一个Java论坛上看到是这样解决的:
import java.lang.reflect.Array;

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

private final T[] array;

}
但我不懂发生了什么。有人能帮我吗?

回答:

在回答之前,我得问你一个问题,你的GetSet是"checked"还是"unchecked"?
什么意思呢?
Checked的话,是强类型。GetSet明确地知道包含了什么类型的对象。
比如,当要传递不是E类型的实参时,它的构造器会被Class引数明确地调用,方***抛出一个异常。参阅Collections.checkedCollection。
在这种情况下,你应该这样写:
public class GenSet {

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

}
Unchecked的话:弱类型。实际上要传递任何对象的实参时
是没有类型检查的。
在这种情况下,你应当这样写:
public class GenSet {

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

}
注意数组里的元素类型应当是可擦除的形参。
public class GenSet { // 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一个有名,存心,不足的泛型:它通过使用
erasure实现,所以“泛型”类在运行时创建是不知道它的实参类型的,
所以不能提供类型安全,除非某些明确的机制(比如类型检查)已经实现了。