泛型
泛型类
public class Person<T>{
private T data;
public T get(){
return data;
}
public void add(T data){
this.data = data;
}
}
如果希望限定传入这个Person的泛型形参为某个类型或者他的子类:
public class Person<T extends Number>{
private T data;
public T get(){
return data;
}
public void add(T data){
this.data = data;
}
}
//这样是用这个Person类的时候就必须是Number类或者其子类
泛型方法:
泛型方法的定义格式
[public|private] <K,V> 返回值 方法名(K k,V v){}
类型推断:
用户不需要传入一个类型参数,编译器会自动推断参数的类型来决定泛型的类型,原则是能是赋值正确的最具体的泛型类型
public <T> abc(T a,T b){}
abc(32,32.0); // 此时T被编译器推断为Double类型
abc(32,new Object()); // 此时T被推断为Object
泛型限定:
通配符:
1. <?>
2. add方法会出错,get方法获取的对象没有编译时类型
public void testGet(List<?> list){
String str = list.get(0);//出错,因为返回值没有编译时类型
System.out.println(str);
}
public void testAdd(List<?> list){
list.add(new Object());//出错,因为你永远不知道传入的list是什么类型,所以给他添加一个具体的类型都是不对的
System.out.println(str);
}
上限限定:
public void upperBound(List<? extends Date> list){
Date date = new Date();
list.add(date);//这一句话会出错,原因是传入的list可能泛型参数为Date的子类,当把一个父类Date赋值给子类就会导致出错
Date d = list.get(0); //这句话不会出错,因为我已经知道list里面的元素类型一定是Date类型或者其子类,
//所以把一个Date类型或者其子类的对象赋值给Date类型绝对不会出问题
}
下限限定:
public void lowerBound(List<? super TimeStamp> list){
TimeStamp ts = list.get(0);//这一句话会出错,原因是传入的list可能泛型参数为TimeStamp的父类,哪父类的对象赋值给子类当然是不行
list.add(new TimeStamp());//这句话是对的,因为已知add的形式参数类型肯定是TimeStamp及其父类,把一个子类对象赋值给父类当然可行
}
易错点:
泛型之间没有继承关系List<String> ls = new ArrayList<Object>(); // 这是不可以的
但是List<String> ls = new ArrayList<String>(); //是可以的
List<Object> lo = new ArrayList<Objct>();
lo.add("hello world"); //这也是没有错误的,因为add的参数类型是Object
List<String> ls = new ArrayList<String>();
lo.add(new Object()); //这会粗错