类型参数有五种不同的情况,只出现一次需要考虑。
1)一旦进入返回类型位置:
1.a) 返回类型是类型变量
public <T extends Something> T getThing();
这应该是一个危险信号:调用者可以任意选择预期的返回类型,而被调用者无法知道选择的类型。换句话说,实现不能保证返回的值将是指定的返回类型,除非它 (a) 永远不会返回,(b) 总是抛出异常或 (c) 总是返回 null。在所有这些情况下,返回类型恰好完全无关。
(如果代码非常“动态”,我个人不介意这样的方法。也就是说,无论如何你都冒着类转换异常的风险,并且方法边界仍然足够早,可以引发异常。一个很好的例子是反序列化。调用该方法的各方都必须知道并理解这一点..)
1.b) 类型变量包含在返回类型中,但不包含返回类型本身
非常普遍和有效。示例是番石榴中的各种工厂方法,例如Lists.newArrayList()
.
2)一旦进入参数类型位置:
2.a) 简单类型参数
public static <E> void shuffle(List<E> list);
请注意,实现实际上需要类型参数才能对元素进行洗牌。尽管如此,调用者不必为此烦恼。您可以编写一个“捕获”通配符的内部辅助方法:
public static void shuffle(List<?> list) {
shuffleWithCapture(list);
}
private static <E> void shuffleWithCapture(List<E> list) {
// implementation
}
2.b) 具有多个边界的类型参数
public static <T extends Foo & Bar> void foo(T);
public static <E extends Foo & Bar> void foo(List<E>);
由于 Java 不允许在除类型参数范围之外的任何地方使用交集类型,因此这是表达这些签名的唯一方法。
2.c) 类型参数绑定包含类型参数变量
public static <T extends Comparable<? super T>> void sort(List<T> list);
为了表示列表的元素必须相互比较,需要为它们的类型命名。没有办法消除此类类型参数。