泛型本质

Author JYBlog
Email jcAuthor@163.com
本博客GitHub开源(jcNaruto/JYBlog)
JYBlog每周五,周三公众号等多平台同步更新,欢迎讨论交流~

  • 泛型(generics, genericity)又称为参数化类型(parameterized types)或模板(templates) 。

  • 是一种通用的复用技术,不同的语言有不同的实现。

  • 泛型在Java语言中通过编译器擦除实现,通过编译期的检查,Java程序也变得更加安全

1.概述

  1. 泛型是一种通用的技术,是代码模板、是一种复用技术(区别于继承), 当你运用泛型时,你将拥有许多不同的类型,并得以相同的演算法(如排序、搜寻)作用在它们身上 。

  2. 在Java中,泛型也是一种编译时类型安全检测机制,你可以在编译期强制检查集合中的元素是否都为同一类型,也可以通过泛型通配符强制参数要实现某些接口(sort方法要实现Comparable原理)。

ArrayList实际上就是会自动扩容的Object[],此时假设放入其中是String类型,取出来时可以强制转换为String,但如果强制转化为其他就会抛出 ClassCastException 。

加入泛型之后,只能存入String类型,也只能取出String类型,如果尝试加入或者取出其他类型,编译器会直接报错,避免了运行期的异常

public class ArrayList ...{
    private Object[] array;
    private int size;
    public void add(Object e) {...}
    ...
}
arrayList.add("123");
Object obj = arrayList.get(0);
String obj = (String)arrayList.get(0);
//ClassCastException
Integer obj = (Integer)arrayList.get(0);

编写了一套排序工具类,方法中间是排序的逻辑,有多个类都符合这个排序逻辑,不用为每个类都写一个排序类,直接用泛型复用。

public class SortUtil{
    public static <K> void mySort(K k){
        xxxxx
    }
}
SortUtil.mySort(new Integer());
SortUtil.mySort(new Double());

2.泛型实现原理

前面提过泛型是一种通用的复用技术,不同的语言有不同的实现。在Java中使用了擦拭法( Type Erasure )

擦拭法的含义就是所有的工作都是编译器做的,jvm对泛型一无所知。

这是程序员和编译器看到的代码

public class JYBlog<T>{
    private T t;
    public void set(T t){
        xxxx
    }
    public T get(){
        xxxxx
    }
}
JYBlog<Integer> test = new JYBlog<>();
test.set(123);
Integer xx = test.get();

这是虚拟机看到的代码

public class JYBlog{
    private Object obj;
    public void set(Object t){
        xxxx
    }
}
JYBlog test = new JYBlog();
test.set(123);
Integer xx = (Integer)test.get();

也就是说编译器在编译时会检查是否符合泛型要求,如果符合的话会让T类型指向Object并且在获取泛型的时候进行一定安全的强制类型转换,虚拟机对此一无所知。

3.擦拭法局限

  1. T不能是基本类型,Object类型无法持有基本类型 ,不然编译报错。

  2. 因为最终虚拟机实际是用T指向了Object,也就是说JYBlog<Integer>和JYBlog<Double>在虚拟机层面都是JYBlog,只不过编译器在getT的时候安全的强制转换了,JYBlog<Integer>.class和JYBlog<Double>.class是同一个,此外编译器还规定,条件表达式中无法判断带泛型的class,因为都是同一个。

  3. 在模板代码中不能实例化T,因为擦拭之后变为new Object();这样就丢失了具体的T。如果模板代码中真的要实例化T可以用反射

    public class JYBlog<T>{
     private T t;
        public JYBlog(Class<T> clazz) {
            T = clazz.newInstance();
        }
    }
    
  4. 泛型方法要防止重复定义方法, 擦拭后就会复写Object的equals,编译器会报错一旦擦拭之后形成复写

public boolean equals(T obj);
  1. 因为擦拭法,我们无法从JYBlog<Integer> y,y实例中获取Integer但是子类可以获取父类的泛型类型T。

4.泛型类、方法、接口

4.1泛型类、泛型方法

将泛型定义在类名后,在实例方法,属性中可以直接使用,不需要定义,但是静态方法上的泛型需要在静态方法上声明,不能直接使用 。程序员在使用该类时,根据不同情况传入不同类型。

public class SortUtil<T>{
    private T t;
    public void set(T t){
        xxxx
    }
    public static <K> void mySort(K k){
        xxxxx
    }
}
SortUtil.mySort(xxx);
new SortUtil<xxx>();

注:<T>,<E>,<K,V>中那些字母没有实际的意思,你可以替换为你想用的字符,但是一般习惯集合中用E,表示元素,其他用T,但是<?>中?代表了通配符中的一种很少使用。

4.2泛型接口

public interface Comparable<T> {
    int compareTo(T o);
}

注:接口是规范,泛型接口的意义就是该规范以后会被不同的类所遵守。

5.通配符和PECS原则

注意:因为擦拭法,所以JYBlog<Integer>和JYBlog<Number>不是子类

void get(ArrayList<Integer> list){
    Number = list.get();
}
add(new ArrayList<Integer>());
//complie error
add(new ArrayList<Double>());

但是从add方法内部可以看出get之后确实可以指向Number,但是由于编译器泛型检查严格,所以禁止这样做,但是我们不能要为每个泛型参数都编写一个方法,因此可以使用extends通配符修饰方法参数,表示可以传入T类型为Number子类的参数

5.1extends通配符

void get(ArrayList<? extends Integer> list){
    Number = list.get();
    list.add(Double);
}
add(new ArrayList<Integer>());
add(new ArrayList<Double>());

此时发现get方法内部list.add一定会出错,因为传入的是Integer泛型的list,无法add double ,因此当extends(上界通配符修饰)修饰方法参数的时候就限制方法内不能修改通配符修饰的参数(null)可以,表示由编译器保证方法内部代码对于参数只能读,不能写。

注:

  1. extends通配符也可以修饰类,public class JYBlog<T extends Number> { ... },这样做会限制T的类型必须是Number或其子类。观看sort的源码就发现sort中通过extends通配符在编译期就要求排序集合的T类型必须实现Comparable接口。

  2. <? extends String>中表示方法参数中的T必须是必须是String的子类或者本身,但是String源码中final修饰class无法继承。也就是参数的泛型必须是String本身。

5.2 super通配符

void set(ArrayList<? super Integer> list,Integer x,Integer y){
    list.add(x);
    //除了返回Object会compile error
    list.get();
}
  1. super通配符修饰方法参数的时候表示,参数的T类型必须是指定类型本身或者其父类。
  2. 在super通配符修饰方法参数修饰的方法内部,执行对应的get方法会编译报错(除了返回Object), 表示由编译器保证方法内部代码对于参数只能写,不能读。
  3. super通配符无法修饰类

5.3 PECS原则

extends,super通配符是由编译器保证的允许读不允许写(null除外),另一个是允许写不允许读 (Object除外),为了方便记忆可以product生产者要读取元素出来使用Extends,消费者要写入使用Super通配符。

5.4 通配符总结

通配符可以限定泛型中的T类型的上界或者下界,如果修饰方法参数还可以在编译期保证方法内只进行读写操作中的一种。

5.5无界通配符

刚刚修饰在方法参数上的上下界通配符其实是限定了传入的参数的T类型的上下界,如果想要指定T可以为任何类型就可以使用<?>,但是方法内部不能读也不能写,只能做一些和null相关的动作。

另外值得注意的是JYBlog<?>是所有JYBlog<T>的超类

6.补充

补充部分待研究到具体章节时会归来补充

  1. 反射中大量用到泛型
  2. 泛型数组存在的意义是什么,数组编译器会检查类型安全,且只能存一种类型,泛型的安全和复用特性完全不需要啊
  3. 同时使用泛型和可变参数时需要特别小心
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,378评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,356评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,702评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,259评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,263评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,036评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,349评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,979评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,469评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,938评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,059评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,703评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,257评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,262评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,485评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,501评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,792评论 2 345