一、引言
在学习集合的时候我们会发现一个问题,将一个对象丢到集合中后,集合并不记住对象的类型,统统都当做Object处理,这样我们取出来再使用时就得强制转换类型,导致代码臃肿,而且加入集合时都是以Object,没做类型检查,那么强制转换就容易出错,泛型的诞生就是为解决这些问题。
二、
ArrayListlist = new ArrayList ();
这是Java 7之前的写法,很明显构造器上面的泛型没有必要,现在推荐以下写法:
ArrayListlist = new ArrayList<>();
既然我已经指定了类型,那么添加时只能添加Integer,并且使用时可以直接当做Integer使用
System.out.println(list.get(2)+3);
这种参数化类型就是泛型,泛型就是允许在定义类、接口、方法时使用类型形参,这个参数形参将在申明变量、创建对象、调用方法时动态指定。
三、
public interface Listextends Collection public interface Map
List接口定义时指定了一个类型形参,Map接口定义了两个类型形参。接口定义了形参之后,在接口中形参就可以当做一种类型来使用,那么其中的方法就可以使用类型形参
boolean add(E e);
这种方式其实也是一种代码复用,我们通过类型形参,高度的将参数抽象,而不需要每种类型都去重新定义类,只要在使用时确定类型即可。我们也可以自定义泛型类
public class WebResult{ //使用T类型形参定义实例变量 private T data; public WebResult() { } //使用T类型形参构造对象 public WebResult(T data) { this.data = data; } public void setData(T data) { this.data = data; } public T getData() { return this.data; } public static void main(String[] args) { WebResult webResult = new WebResult<>("返回一个String对象"); System.out.println(webResult.getData()); WebResult webResult1 = new WebResult<>(10); System.out.println(webResult1.getData()); } }
四、
public static void main(String[] str){ ArrayListarrayList=new ArrayList(); test(arrayList); } public static void test(List
这段代码会出现编译错误,因为List
将方法修改:
public static void test(List> test){ for (int i = 0; i) { System.out.println(test.get(i)); } }
这样便可以顺利编译,我们再加上这段代码:
public static void main(String[] str){ ArrayListarrayList=new ArrayList(); test(arrayList); List> strings = arrayList; strings.add("abc"); }
这里我们可以将arrayList给strings,按说这个时候是不能赋值的,因为List不知道类型参数的值,这是编译器作用,可以进行类型推理,但是后面的strings.add("abc")是不能通过编译的,编译器不能对 List 的类型参数作出足够严密的推理,以确定将 String 传递给 List.add() 是类型安全的。所以编译器将不允许这么做。
4.1 设置通配符上限
List>这种方式,通配的是所有的类型,很多时候我们可以确定是哪一类对象可以添加进去,我们只希望它代表某一类泛型的父类,这个时候我们可以设置通配符的上限。
//动物类 public abstract class Animal { public abstract void say(); } public class Cat extends Animal { @Override public void say() { System.out.println("喵喵"); } } public class Dog extends Animal { @Override public void say() { System.out.println("旺旺"); } }
这个时候我们就限定了上限
public static void test1(List extends Animal> animals) { for (int i = 0; i < animals.size(); i++) { animals.get(i).say(); } }
我们也可以直接在定义类型形参的时候设置上限
public class WebResultextends Animal> {
//将src中的集合复制到dest,并返回最后一个值 public staticT copy(Collection dest, Collection extends T> src) { T last = null; for (T ele : src) { last = ele; dest.add(ele); } return last; }
功能比较很简单,将src中的集合复制到dest,并返回src最后一个值
Listln = new ArrayList<>(); List li = new ArrayList<>(); //编译出错,类型不确定 Integer last = copy(ln, li);
这个时候出错,因为虽然我们知道返回的值一定是Integer,但是由于copy方法的返回值并不是,所有相当于我们在复制的过程中丢失了src的类型,如果我们想定义约束关系使得返回值明确即:dest集合元素类型与src的关系要么相同要么是其父类,为了表示这种约束关系,引入了 super T> 这个通配符表示它必须是T本身或者T的父类。
//将src中的集合复制到dest,并返回最后一个值 public staticT copy(Collection super T> dest, Collection extends T> src) { T last = null; for (T ele : src) { last = ele; dest.add(ele); } return last; }
五、
staticvoid arrayToList(T[] a, List list) { for (T o : a) { list.add(o); } }
调用如下:
Object[] objects = new Object[10]; List
这里可以看出泛型方法跟类型通配符的功能有点类似,其实在大部分情况下我们可以用泛型方法代替类型通配符。
泛型方法允许类型形参被用来表示方法的一个或者多个参数之间的依赖关系,或者说与返回值之间的关系,如果没有这种关系,我们就不使用泛型方法。
六、擦除与转换
当把一个具有泛型信息的对象赋给一个没有泛型信息的变量时,所有的类型信息就都丢掉了,比如List
public class WebResultextends Number> { //使用T类型形参定义实例变量 private T data; public WebResult() { } //使用T类型形参构造对象 public WebResult(T data) { this.data = data; } public void setData(T data) { this.data = data; } public T getData() { return this.data; } public static void main(String[] args) { WebResult webResult1 = new WebResult<>(10); System.out.println(webResult1.getData()); WebResult a = new WebResult<>(20); WebResult b = a; //已经擦除了泛型,只能按最高类型Object //Integer bData = b.getData(); Object object=b.getData(); } }
原本的泛型类上限是Number,而当把a赋给擦除泛型的b对象时,编译器失去了推断能力,只能把其当做Objec来处理。
而当一个List转成泛型对象是java是允许的
ListintegerList = new ArrayList<>(); List stringList = integerList; //允许直接将list对象转换给 List strings = stringList; //直接获取数据会出现错误,因为转换不成功 System.out.println(stringList.get(0));