版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/lastsweetop/article/details/83010344
我们在之前说过明确指定了泛型类型的参数之后,会把参数限制的很严格,通配符的作用就是放宽这种限制,有上限有界通配符 <? extends A>
,下限有界通配符 <? extends B>
,无界通配符 <?>
。
上限有界是in变量,而下限有界是out变量,
虽然通配符可以使用在字段,本地变量和方法中,但是类和外界的交互主要是方法,就限定的意义本身还是主要通过方法来体现,因此我们主要说方法。
考虑到通配符的一个弱点,遍历和移除肯定是没问题的,就是不能对通配符修饰的类型进行添加元素操作(可以添加null)。
比如:
void swapFirst(List<? extends Number> l1, List<? extends Number> l2) { Number temp = l1.get(0); l1.set(0, l2.get(0)); // expected a CAP#1 extends Number, // got a CAP#2 extends Number; // same bound, but different types l2.set(0, temp); // expected a CAP#1 extends Number, // got a Number }
编译阶段居然会报错,编译器无法确认类型是正确的,因为你输入的具体类型可能是 Number
的子类哦。
这种情况下还是写泛型来的舒服,也能达到同样的效果
<E extends Number> void swapSecond(List<E> l1, List<E> l2) { E temp = l1.get(0); l1.set(0, l2.get(0)); l2.set(0, temp); }
这样就可以正常工作了。
上面是上限的可以用这种替换,如果是无界的可以直接用 T
表示,至于下限有界的真没办法。
那么可以探讨一下通配符的意义何在,为了暴露方法时更为一目了然?
因此,私以为能用泛型的情况下还是不用通配符了。