泛型和反射

调用ClassgetSuperclass()方法返回的Class类型是Class<? super T>

  1. Class<? super String> sup = String.class.getSuperclass();

构造方法Constructor<T>也是泛型:

  1. Class<Integer> clazz = Integer.class;
  2. Constructor<Integer> cons = clazz.getConstructor(int.class);
  3. Integer i = cons.newInstance(123);

我们可以声明带泛型的数组,但不能用new操作符创建带泛型的数组:

  1. Pair<String>[] ps = null; // ok
  2. Pair<String>[] ps = new Pair<String>[2]; // compile error!

必须通过强制转型实现带泛型的数组:

使用泛型数组要特别小心,因为数组实际上在运行期没有泛型,编译器可以强制检查变量ps,因为它的类型是泛型数组。但是,编译器不会检查变量arr,因为它不是泛型数组。因为这两个变量实际上指向同一个数组,所以,操作arr可能导致从ps获取元素时报错,例如,以下代码演示了不安全地使用带泛型的数组:

  1. Pair[] arr = new Pair[2];
  2. Pair<String>[] ps = (Pair<String>[]) arr;
  3. ps[0] = new Pair<String>("a", "b");
  4. // ClassCastException:
  5. Pair<String> p = ps[1];
  6. String s = p.getFirst();

要安全地使用泛型数组,必须扔掉arr的引用:

  1. @SuppressWarnings("unchecked")
  2. Pair<String>[] ps = (Pair<String>[]) new Pair[2];

上面的代码中,由于拿不到原始数组的引用,就只能对泛型数组ps进行操作,这种操作就是安全的。

  1. Pair[] arr = new Pair[2];
  2. Pair<String>[] ps = (Pair<String>[]) arr;
  3. System.out.println(ps.getClass() == Pair[].class); // true
  4. String s1 = (String) arr[0].getFirst();
  5. String s2 = ps[0].getFirst();

所以我们不能直接创建泛型数组T[],因为擦拭后代码变为Object[]

必须借助Class<T>来创建泛型数组:

  1. return (T[]) Array.newInstance(cls, 5);
  2. }

我们还可以利用可变参数创建泛型数组T[]

  1. @SafeVarargs
  2. static <T> T[] asArray(T... objs) {
  3. return objs;
  4. }
  5. }
  6. String[] ss = ArrayHelper.asArray("a", "b", "c");
  7. Integer[] ns = ArrayHelper.asArray(1, 2, 3);

在上面的例子中,我们看到,通过:

  1. static <T> T[] asArray(T... objs) {
  2. return objs;

似乎可以安全地创建一个泛型数组。但实际上,这种方法非常危险。以下代码来自《Effective Java》的示例:

直接调用asArray(T…)似乎没有问题,但是在另一个方法中,我们返回一个泛型数组就会产生ClassCastException,原因还是因为擦拭法,在pickTwo()方法内部,编译器无法检测K[]的正确类型,因此返回了Object[]

如果在方法内部创建了泛型数组,最好不要将它返回给外部使用。

更详细的解释请参考《Effective Java》“Item 32: Combine generics and varargs judiciously”。

部分反射API是泛型,例如:Class<T>Constructor<T>

可以声明带泛型的数组,但不能直接创建带泛型的数组,必须强制转型;

可以通过Array.newInstance(Class<T>, int)创建T[]数组,需要强制转型;

同时使用泛型和可变参数时需要特别小心。

泛型和反射 - 图1