返回 登录
0

Java装逼三部曲之泛型

阅读2111

原文地址
首先我们先提出两个问题:

  • 什么是泛型?
  • 为什么要使用泛型?
    我们先来看看第一个问题什么是泛型。如果你对Java三大特性中的多态性理解的比较透彻的话,泛型就比较好理解了。多态性表示一个对象具备多种状态。比如说你自己,你首先是一个人,同时你在看这篇帖子说明你是一个程序员,下了班之后你可能有变成了禽兽。所以你具备这人形态,程序员形态,和兽形态,这就是多态,一种事物具备多种状态!而泛型就更加纯粹的多态,他可以是任何一种形态,俗称变态(开个玩笑o(∩_∩)o)。泛型的意思就是可以表示任何一种形态,任何一种数据类型。这就好像情人节你女朋友送你的礼物一样,你先拿到的是一个礼物包装盒,而盒子里面可能是一个钱包,也可能是一个高档打火机,还有可能是一坨屎!盒子里面有可能是任何一种事物,你对此完全不知道,你知道你会收到这样一份礼物,而礼物具体是什么,你全然不知。
 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而不是泛型的话,集合容器使用起来就没那么顺手了。使用泛型能极大的提高我们程序的灵活性和拓展性,比如我写的万能适配器中 就使用了泛型来进行数据源的通用化处理。从上面两个例子可以看出泛型的使用场景:当数据类型不确定时,或者说需要使用多个数据类型时,泛型则是不错的选择。
泛型的功能远不止于此,有关获取泛型的数据类型,泛型的边界,泛型的通配符等问题,将会在下篇帖子中详解,欢迎继续关注

评论