Java泛型详解(一)

1、什么是泛型

泛型,是“参数化类型”的概念,代码可以适用于多种类型。参数化类型是指把具体的类型参数化,就像方法中使用的参数一样。

    void hello(String str){
        System.out.println("str是个变量参数,String是个具体类型");
    }
    <T> void hello(T str){
        System.out.println("str是个变量参数,T是个类型参数");
    }

2、为什么使用泛型

泛型出现最引人注目的一个原因,是为了创造使用容器类。
        List arrayList=new ArrayList();
        arrayList.add("hello");
        arrayList.add(110);
        for(Object obj:arrayList){
            String s=(String)obj;//java.lang.ClassCastException
        }

有些情况下,我们希望容器能够持有多种类型的对象,但是,大部分情况我们只会使用容器存储一种类型的对象,泛型可以在编译阶段就可以防止错误的发生。

List<String> arrayList=new ArrayList<String>();
arrayList.add(110);//编译时报错
另外,一般的类和方法,只能使用具体的类型。如果要编写可以应用于多种类型的代码,这种刻板的限制就会对代码造成束缚。

return语句只能返回一个对象,但是有很多功能要求一个方法需要返回多个对象,怎么办?
解决方法就是创建一个对象,用它来持有返回的多个对象。可以在每次使需要时,专门创建一个类,而使用泛型则能够一次性解决该问题,同时在编译期间就能确保类型安全。

public class TwoTuple <A,B>{
    public final A a;
    public final B b;
    public TwoTuple(A a,B b) {
        this.a=a;
        this.b=b;
    }
}

例:获取一个学生的信息和导师的姓名。

public class TupleTest{
    public static void main(String[] args) {
        TwoTuple<Student,String> tt=new TwoTuple<Student, String>(new Student(), "张大侠");
    }
}
class Student{}

3、泛型特性

泛型只在编译阶段有效

        List<String> stringArrayList = new ArrayList<String>();
        List<Integer> integerArrayList = new ArrayList<Integer>();
        Class<? extends List> classStringArrayList = stringArrayList.getClass();
        Class<? extends List> classIntegerArrayList = integerArrayList.getClass();
        if(classStringArrayList.equals(classIntegerArrayList)){
            System.out.println("类型相同");
        }

输出:

类型相同

泛型不强制使用,泛型也无法显示用于运行时类型的操作,例如转型、instanceof和new表达式。

    public void testClass(){
        GenricClass<Integer> gci=new GenricClass<Integer>(10086);
//      GenricClass<String> gcs=new GenricClass<String>(10086);
        GenricClass<String> gcs=new GenricClass<String>("hello");
        GenricClass g1=new GenricClass(10086);
        GenricClass g2=new GenricClass("hello");
        GenricClass g3=new GenricClass(66.666);
        GenricClass g4=new GenricClass(true);
        System.out.println(g1.getField());
        System.out.println(g2.getField());
        System.out.println(g3.getField());
        System.out.println(g4.getField());
        if(g2 instanceof GenricClass<?>){//if(g2 instanceof GenricClass<String>) 编译错误
            System.out.println("true");
        }
    }

Java泛型是使用擦除来实现的,意思是当你使用泛型的时候,任何具体的类型信息都被擦除了,你唯一知道的就是你在使用一个对象。List<String> 和List<Integer>在运行时是相同的类型,都被擦除成他们的原生类型,即List。
擦除默认会把类型擦除到Object,如果需要使用具体类的某些特性,需要协助泛型类,给泛型类定个边界。

    class Shape{int length() {return 0;}}
    class Rectangle extends Shape{}
    class Circle extends Shape{}
    class Painter<T extends Shape>{
        private T t;
        public Painter(T t) {this.t=t;}
        public void draw(){
            t.length();
        }
    }

在上面这个例子中,泛型并没有贡献什么好处,只要把T替换成Shape就可以很容易创建出没有泛型的类,这也说明一点:只有当你是用的类型参数比某个具体类型更加通用时,泛型才有帮助。

4、怎么用泛型

泛型有三种使用方式:泛型类、泛型接口、泛型方法。

public interface GenricInterface<T> {
    public T getField();
}

public class GenricClass<T> implements GenricInterface<T>{
    private T field;
    public GenricClass(T f) {field=f;}
    public T getField() {return field;}
//  静态泛型方法无法访问类中定义的类型。必须额外声明
    public static <T> T genricMethod(Class<T> tc) throws InstantiationException, IllegalAccessException{
        T t=tc.newInstance();
        return t;
    }
    public void show1(T t){System.out.println(t.toString());}
    public <E> void show2(E e){System.out.println(e.toString());}
    public <T> void show3(T t){System.out.println(t.toString());}
}

前面说过了泛型无法通过new T( )创建对象实例,Java的解决方案是使用工厂对象来创建新的实例,最便利的工厂对象就是Class对象,可以通过newInstance()来创建该类型新对象。

    public void testGenricMethod(){
        System.out.println("泛型方法测试----------------------------------------------------------");
        try {
            NormalClass nc= GenricClass.genricMethod(NormalClass.class);
        } catch (InstantiationException | IllegalAccessException e ) {
            e.printStackTrace();
        }
        GenricClass<Integer> g1=new GenricClass<Integer>(10086);
        g1.show1(100);
//      g1.show1(100.1);
        g1.show2(100);      g1.show2(100.1);
        g1.show3(100);      g1.show3(100.1);
    }

泛型通配符

    public void testWildcard(){//通配符测试
        System.out.println("通配符测试----------------------------------------------------------");
        GenricClass<Integer> g1=new GenricClass<Integer>(10086);
        GenricClass<Number> g3=new GenricClass<Number>(66.666);
//      show(g1);//参数不匹配        
        show(g3);
        show2(g1);      show2(g3);
        show3(g1);      show3(g3);
//      show4(g1);      show4(g3);//参数不匹配   <?>可理解为所有泛型父类<? extends Object>
    }
    public void show(GenricClass<Number> gc){
        System.out.println("show:"+gc.getField());
    }
    public void show2(GenricClass<? extends Number> gc){
        System.out.println("show2:"+gc.getField());
    }
    public void show3(GenricClass<?> gc){
        System.out.println("show3:"+gc.getField());
    }
    public void show4(GenricClass<Object> gc){
        System.out.println("show4:"+gc.getField());
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,802评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,109评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,683评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,458评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,452评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,505评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,901评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,550评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,763评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,556评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,629评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,330评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,898评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,897评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,140评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,807评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,339评论 2 342

推荐阅读更多精彩内容

  • 泛型是Java 1.5引入的新特性。泛型的本质是参数化类型,这种参数类型可以用在类、变量、接口和方法的创建中,分别...
    何时不晚阅读 3,023评论 0 2
  • 在之前的文章中分析过了多态,可以知道多态本身是一种泛化机制,它通过基类或者接口来设计,使程序拥有一定的灵活性,但是...
    _小二_阅读 676评论 0 0
  • 附上思维导图。这篇博客主要讲了如下知识点。 看完了《Thinking in Java》的第十五章泛型,着实被震了一...
    Happioo阅读 710评论 0 1
  • 关于作者 吉姆·柯明斯,资深的广告专家,在多家机构担任首席战略规划师。 关于本书 本关于如何说服他人的工具书。想要...
    Wells陆阅读 837评论 0 1
  • 没有街头闪烁的霓虹,没有塞满排队的车流,更没有风云激昂的传说。沂蒙一角,蟋蟀略鸣,黑狗稍燥,微风渐起,暮色四...
    fc8711f32b28阅读 163评论 0 1