发布时间:2016-01-11 00:00 来源:soul_code
首先我们先提出两个问题:
什么是泛型? 为什么要使用泛型?public class GrilFriend<T> { private T t; public void buyGift(T t) { this.t = t; } public T getGift() { return t; } public static void main(String[] args) { GrilFriend<Shit> myGrilFriend = new GrilFriend<Shit>(); myGrilFriend.buyGift(new Shit("狗屎")); Shit gift = myGrilFriend.getGift(); System.out.println(gift.name); } } class Shit { final String name; public Shit(String name) { this.name = name; } }
这就是泛型,你永远不知道你女朋友会送你什么礼物,这里的礼物可能是任何一种类型(她可能会真的送你一坨屎)
接下来我们来看我们为什么要使用泛型 ,相信List集合容器,大家都用的滚瓜烂熟了,我们先来看下List的源码
public interface List<E> extends Collection<E> { public boolean add(E object); public boolean addAll(Collection<? Extends E> collection); public E get(int location); }
我们只看这几个我们常用的方法,我们知道List是一个容器,它就是使用了泛型,我们才可以用List来存储任何一种数据类型,有的小伙伴就会说,那用顶级父类Object还不是一样可以实现这样的功能,代码如下:
public interface List extends Collection { public boolean add(Object object); public boolean addAll(Collection<? Extends Object> collection); public Object get(int location); }
确实这样也能实现多种数据的存储,但是有一个很严重的缺陷,当我们用Object来接数据的时候,一旦数据存储到容器中,容器就会忘记存储的数据类型,因为只知道当前对象是Object类型,我们使用起来的代码就会变成这样
List list = new ArrayList(); list.add("hhhh"); list.add("eeee"); list.add("wwww"); for (int I = 0; I < list.size(); i++) { // 每次取数据 都需要进行强转操作 String object = (String) list.get(i); System.out.println(object); } List list2 = new ArrayList(); list2.add("hhhh"); list2.add("eeee"); list2.add("wwww"); //如果这里不小心添加的多种数据类型,编译器不会有任何提示信息 list2.add(1); for (int I = 0; I < list.size(); i++) { // 当使用强转是就会报ClassCastException String object = (String) list.get(i); System.out.println(object); }
如果List使用的是Object而不是泛型的话,集合容器使用起来就没那么顺手了。使用泛型能极大的提高我们程序的灵活性和拓展性,比如我写的万能适配器中 就使用了泛型来进行数据源的通用化处理。从上面两个例子可以看出泛型的使用场景:当数据类型不确定时,或者说需要使用多个数据类型时,泛型则是不错的选择。
泛型的功能远不止于此,有关获取泛型的数据类型,泛型的边界,泛型的通配符等问题,将会在下篇帖子中详解,欢迎继续关注。