上次说了泛型的语法,发现有个问题,就是泛型是什么就得是什么,但这样岂不是程序就得写死了,没有一点可预判性了吗,那么泛型的好处也就光体现在不用进行强制类型转换上了吗?
泛型还给我们提供了一种符号 : ? 这种符号代表通配符,意思是什么都可以,看例子
/**
* 打印任意集合的值
* @param list
*/
public static void printSize(List<?> list){
//不可以执行任何与类型有关的方法
// list.add("what?");
for (Object object : list) {
System.out.println(object);
}
//这样也没错,因为通配符可以指向任何类型的引用
list = new ArrayList<String>();
}
如果这样写一个方法的参数,那么此方法可以接受任何类型参数化的List,正好解决了参数化类型必须相同的尴尬(上一次泛型语法的坑有记录到)。
那么既然可以接受任何类型参数化为什么list.add("what")又不可以呢,正是因为他用了通配符,所以这里有可以传入任何类型的list,加入说传入的是一个Integer的list那么进行add字符串操作不就是错了吗? 所以想这样使用泛型的通配符,编译器是不允许使用它的任何与参数有关的API的,但我们可以使用它的其他API,例如,长度,删除,取值等。
但是也可以将引用指向任何参数化类型的对象,例如list=new ArrayList<String>();这样使用,所以总结一下通配符的作用大体是:
通配符定义的变量主要作为引用来使用,可以指向任何参数化类型的,并且调用起与参数无关的API。
给通配符定义上限或下限:
public static void main(String[] args) {
/**
* 限定上边界
*/
//这样意味着可以指向任何参数是Object以及其子类的list
List<? extends Object> list = new ArrayList<String>();
//如果是这样就不可以,上次已经记录过,因为泛型是不会自动考虑继承关系的,也正是因为这样,通配符的出现解决了这个尴尬
//List<Object> list = new ArrayList<String>();
/**
* 限定下边界
*/
//这意味着可以指向任意参数是FileInputStream以及它的父类的list
List<? super FileInputStream> list2 = new ArrayList<InputStream>();
}
注意: 限定通配符的出现总是要包括自己,比如 ? extends Object 意味 <= Object ,? super FileInPutStream 意味着 >= FileInputStream