泛型中通配符" ?"的使用
使用类型通配符:?
- 比如: List<?> , Map<?,?>
List<?>是List、 List等各种泛型List的父类。 - 读取List<?>的对象list中的元素时,永远是安全的,因为不管list的真实类型是什么,它包含的都是Object。
- 此代码中,将任意元素加入到其中不是类型安全的:↓
Collection<?> c = new ArrayList<String>();
c.add(new Object()); // 编译时错误
因为我们不知道c的元素类型,我们不能向其中添加对象。 add方法有类型参数E作为集
合的元素类型。我们传给add的任何参数都必须是一个未知类型的子类。
- 唯一的例外的是null,它是所有类型的成员:↓
public static void main(String[] args) {
List<?> list = null;
list = new ArrayList<String>();
list = new ArrayList<Double>();
// list.add(3);//编译不通过
list.add(null);
List<String> l1 = new ArrayList<String>();
List<Integer> l2 = new ArrayList<Integer>();
l1.add("中国");
l2.add(15);
read(l1);
read(l2);
}
public static void read(List<?> list) {
for (Object o : list) {
System.out.println(o);
}
}
- 另一方面,我们可以调用get()方法并使用其返回值。返回值是一个未知的类型,但是我们知道,它总是一个Object。
通配符的使用:注意点
- 注意点1:编译错误:不能用在泛型方法声明上,返回值类型前面<>不能使用?
public static <?> void test(ArrayList<?> list){
}
- 注意点2:编译错误:不能用在泛型类的声明上
class GenericTypeClass<?>{
}
- 注意点3:编译错误:不能用在创建对象上,右边属于创建集合对象
ArrayList<?> list2 = new ArrayList<?>();
通配符的使用:有限制的通配符
- 通配符指定上限
上限extends:使用时指定的类型必须是继承某个类,或者实现某个接口,即<= - 通配符指定下限
下限super:使用时指定的类型不能小于操作的类,即>= - 举例:
- <? extends Number> (无穷小 , Number] 只允许泛型为Number及Number子类的引用调用
- <? super Number> [Number , 无穷大) 只允许泛型为Number及Number父类的引用调用
- <? extends Comparable> 只允许泛型为实现Comparable接口的实现类的引用调用
public static void printCollection3(Collection<? extends Person> coll) {
//Iterator只能用Iterator<?>或Iterator<? extends Person>.why?
Iterator<?> iterator = coll.iterator();
while (iterator.hasNext()) {
System.out.println(iterator.next());
}
}
public static void printCollection4(Collection<? super Person> coll) {
//Iterator只能用Iterator<?>或Iterator<? super Person>.why?
Iterator<?> iterator = coll.iterator();
while (iterator.hasNext()) {
System.out.println(iterator.next());
}
}