JAVA 基础 - 并不神奇的泛型

前言

前阵子给公司新人培训Java 基础相关的一些点,系统整理了一下泛型相关的知识点。特来分享一下。希望能让一些对泛型不熟悉的同学完全掌握Java 泛型的相关知识点。

开始之前,先给大家来一道测试题。

List<String> strList = new ArrayList<String>();
List<Integer> integerList = new ArrayList<Integer>();
        
System.out.println(strList.getClass() == integerList.getClass());

请问,上面代码最终结果输出的是什么?熟悉泛型的同学应该能够答出来,而对泛型有所了解,但是了解不深入的同学可能会答错。

content

  • 泛型概述
    • why 泛型
    • 泛型的作用
  • 泛型的定义和使用
    • 泛型类
    • 泛型方法
    • 泛型接口
  • 通配符 ?
    • 无界通配符
    • 上限通配符
    • 下限通配符
  • 类型擦除

带着问题

  1. Java中的泛型是什么 ? 使用泛型的好处是什么?
  2. 什么是泛型中的限定通配符和无界通配符 ?
  3. 你可以把List<String>传递给一个接受List<Object>参数的方法吗?
  4. Java的泛型是如何工作的 ? 什么是类型擦除 ?

一、泛型概述

最早的“泛型编程”的概念起源于C++的模板类(Template),Java 借鉴了这种模板理念,只是两者的实现方式不同。C++ 会根据模板类生成不同的类,Java 使用的是类型擦除的方式。

1.1 why 泛型?

Java1.5 发行版本中增加了泛型(Generic)。

有很多原因促成了泛型的出现,而最引人注意的一个原因,就是为了创建容器类。

-- 《Java 编程思想》

容器就是要存放要使用的对象的地方。数组也是如此,只是相比较的话,容器类更加的灵活,具有更多的功能。所有的程序,在运行的时候都要求你持有一大堆的对象,所以容器类算得上最需要具有重用性的类库之一了。

看下面这个例子,


public class AutoMobile {
}

/**
 * 重用性不好的容器类
 */
public class Holder1 {

    private AutoMobile a;

    public Holder1(AutoMobile a) {
        this.a = a;
    }
    //~~
}

/**
 * 想要在java5 之前实现可重用性的容器类
 * @author Richard_yyf
 * @version 1.0 2019/8/29
 */
public class Holder2 {

    private Object a;

    public Holder2(Object a) {
        this.a = a;
    }

    public Object getA() {
        return a;
    }

    public void setA(Object a) {
        this.a = a;
    }

    public static void main(String[] args) {
        Holder2 h2 = new Holder2(new AutoMobile());
        AutoMobile a = (AutoMobile) h2.getA();
        h2.setA("Not an AutoMobile");
        String s = (String) h2.getA();
        h2.setA(1);
        Integer x = (Integer) h2.getA();
    }
}



/**
 * 通过泛型来实现可重用性
 * 泛型的主要目的是指定容器要持有什么类型的对象
 * 而且由编译器来保证类型的正确性
 *
 * @author Richard_yyf
 * @version 1.0 2019/8/29
 */
public class Holder3WithGeneric<T> {

    private T a;

    public Holder3WithGeneric(T a) {
        this.a = a;
    }

    public T getA() {
        return a;
    }

    public void setA(T a) {
        this.a = a;
    }

    public static void main(String[] args) {
        Holder3WithGeneric<AutoMobile> h3 = new Holder3WithGeneric<>(new AutoMobile());
        // No class cast needed
        AutoMobile a = h3.getA();
    }
}

通过上述对比,我们应该可以理解类型参数化具体是什么个意思。

在没有泛型之前,从集合中读取到的每一个对象都需要进行转换。如果有人不小心插入了类型错误的对象,在运行时的转换处理就会出错。这显然是不可忍受的。

泛型的出现,给Java带来了不一样的编程体验。

1.2 泛型的作用

  1. 参数化类型。与普通的 Object 代替一切类型这样简单粗暴而言,泛型使得数据的类别可以像参数一样由外部传递进来。它提供了一种扩展能力。它更符合面向抽象开发的软件编程宗旨。
  2. 类型检测。当具体的类型确定后,泛型又提供了一种类型检测的机制,只有相匹配的数据才能正常的赋值,否则编译器就不通过。所以说,它是一种类型安全检测机制,一定程度上提高了软件的安全性防止出现低级的失误。
  3. 提高代码可读性。不必要等到运行的时候才去强制转换,在定义或者实例化阶段,因为 Holder<AutoMobile>这个类型显化的效果,程序员能够一目了然猜测出这个容器类持有的数据类型。
  4. 代码重用。泛型合并了同类型对象的处理代码,使得代码重用度变高。

二、泛型的定义和使用

泛型按照使用情况可以分为 3 种。

  1. 泛型类
  2. 泛型方法
  3. 泛型接口

2.1 泛型类

  • 概述:把泛型定义在类上

  • 定义格式:

    public class 类名 <泛型类型1,...> {
        ...
    }
    
  • 注意事项:泛型类型必须是引用类型(非基本数据类型)

类型参数 规范(约定俗称)

尖括号 <>中的 字母 被称作是类型参数,用于指代任何类型。我们常看到<T> 的写法,事实上,T 只是一种习惯性写法,如果你愿意。你可以这样写。

public class Test<Hello> {
    Hello field1;
}

但出于规范和可读性的目的,Java 还是建议我们用单个大写字母来代表类型参数。常见的如:

  • T 代表一般的任何类。
  • E 代表 Element 的意思,或者 Exception 异常的意思。
  • K 代表 Key 的意思。
  • V 代表 Value 的意思,通常与 K 一起配合使用。
  • S 代表 Subtype 的意思

2.2 泛型方法

  • 概述:把泛型定义在方法上

  • 定义格式:

    public <泛型类型> 返回类型 方法名(泛型类型 变量名) {
        ...
    }
    
  • 注意事项:

    • 这里的<T> 中的T被称为类型参数,而方法中的 T 被称为参数化类型,它不是运行时真正的参数。
    • 方法声明中定义的形参只能在该方法里使用,而接口、类声明中定义的类型形参则可以在整个接口、类中使用

泛型类和泛型方法共存的现象

/**
 * 泛型类与泛型方法的共存现象
 * @author Richard_yyf
 * @version 1.0 2019/8/29
 */
public class GenericDemo2<T> {

    public  void testMethod(T t){
        System.out.println(t.getClass().getName());
    }
    public  <T> T testMethod1(T t){
        return t;
    }

    public static void main(String[] args) {
        GenericDemo2<String> t = new GenericDemo2<>();
        t.testMethod("generic");
        Integer integer = 1;
        Integer i = t.testMethod1(integer);

    }
}

泛型方法始终以自己定义的类型参数为准

当然,现实场景下千万不要去作死写出这么难以阅读的代码。

2.3 泛型接口

泛型接口和泛型类差不多。

  • 泛型接口概述:把泛型定义在接口

  • 定义格式:

    public interface 接口名<泛型类型> {
        ...
    }
    

Demo

public interface GenericInterface<T> {

    void show(T t);
}

public class GenericInterfaceImpl<String> implements GenericInterface<String>{

    @Override
    public void show(String o) {

    }
}

三、 通配符 ?

除了用 <T>表示泛型外,还有 <?>这种形式。 被称为通配符。

为什么要引进这个概念呢?先来看下下面的Demo.

public class GenericDemo2 {

    class Base{}

    class Sub extends Base{}

    public void test() {
        // 继承关系
        Sub sub = new Sub();
        Base base = sub;
        List<Sub> lsub = new ArrayList<>();
        // 编译器是不会让下面这行代码通过的,
        // 因为 Sub 是 Base 的子类,不代表 List<Sub>和 List<Base>有继承关系。
        List<Base> lbase = lsub;
    }
}

在现实编码中,确实有这样的需求,希望泛型能够处理某一范围内的数据类型,比如某个类和它的子类,对此 Java 引入了通配符这个概念。

所以,通配符的出现是为了指定泛型中的类型范围

通配符有 3 种形式。

  1. <?>被称作无限定的通配符
  2. <? extends T>被称作有上限的通配符
  3. <? super T>被称作有下限的通配符

3.1 无界通配符 <?>

无限定通配符经常与容器类配合使用,它其中的 ? 其实代表的是未知类型,所以涉及到 ? 时的操作,一定与具体类型无关。

// Collection.java
public interface Collection<E> extends Iterable<E> {
   
    boolean add(E e);
}

public class GenericDemo3 {
    /**
     * 测试 无限定通配符 <?>
     * @param collection c
     */
    public void testUnBoundedGeneric(Collection<?> collection) {
        collection.add(123);
        collection.add("123");
        collection.add(new Object());

        // 你只能调用 Collection 中与类型无关的方法
        collection.iterator().next();
        collection.size();
    }
}

无需关注 Collection 中的真实类型,因为它是未知的。所以,你只能调用 Collection 中与类型无关的方法。

有同学可能会想,<?>既然作用这么渺小,那么为什么还要引用它呢? �

个人认为,提高了代码的可读性,程序员看到这段代码时,就能够迅速对此建立极简洁的印象,能够快速推断源码作者的意图。

(用的很少,但是要理解)

为了接下去的说明方便,先定义一下几个类。

    class Food {}

    class Fruit extends Food {}

    class Apple extends Fruit {}

    class Banana extends Fruit {}

    // 容器类
    class Plate<T> {
        private T item;

        public Plate(T item) {
            this.item = item;
        }

        public T getItem() {
            return item;
        }

        public void setItem(T item) {
            this.item = item;
        }
    }

3.2 上限 通配符 <? extends T>

<?>代表着类型未知,但是我们的确需要对于类型的描述再精确一点,我们希望在一个范围内确定类别,比如类型 T 及 类型 T 的子类都可以放入这个容器中。

什么是上界

在这个体系中,上限通配符 Plate<? extends Fruit> 覆盖下图中蓝色的区域。

image.png

副作用

边界让Java不同泛型之间的转换更容易了。但不要忘记,这样的转换也有一定的副作用。那就是容器的部分功能可能失效。

 public void testUpperBoundedBoundedGeneric() {
       Plate<? extends Fruit> p = new Plate<>(new Apple());

       // 不能存入任何元素
        p.setItem(new Fruit()); // error
        p.setItem(new Apple()); // error

        // 读出来的元素需要是 Fruit或者Fruit的基类
        Fruit fruit = p.getItem();
        Food food = p.getItem();
//        Apple apple = p.getItem();
    }

<? extends Fruit>会使往盘子里放东西的set( )方法失效。但取东西get( )方法还有效。比如下面例子里两个set()方法,插入Apple和Fruit都报错。

原因是编译器只知道容器内是Fruit或者它的派生类,但具体是什么类型不知道。可能是Fruit?可能是Apple?也可能是Banana,RedApple,GreenApple?

如果你需要一个只读容器,用它来produce T,那么使用<? extends T> 。

3.3 下限通配符 <? super T>

相对应的,还有下限通配符 <? super T>

什么是下界

对应刚才那个例子,Plate<? super Fruit>覆盖下图中红色的区域。

image.png

副作用

 public void testLowerBoundedBoundedGeneric() {
//        Plate<? super Fruit> p = new Plate<>(new Food());
        Plate<? super Fruit> p = new Plate<>(new Fruit());

        // 存入元素正常
        p.setItem(new Fruit());
        p.setItem(new Apple());

        // 读取出来的东西,只能放在Object中
        Apple apple = p.getItem(); // error
        Object o = p.getItem();
    }

因为下界规定了元素的最小粒度的下限,实际上是放松了容器元素的类型控制。既然元素是Fruit的基类,往里面存比Fruit粒度小的类都可以。但是往外读取的话就费劲了,只有所有类的基类Object可以装下。但这样一来元素类型信息就都丢失了。

3.4 PECS 原则

PECS - Producer Extends Consumer Super

  • “Producer Extends” – 如果你需要一个只读容器,用它来produce T,那么使用<? extends T> 。
  • “Consumer Super” – 如果你需要一个只写容器,用它来consume T,那么使用<? super T>。
  • 如果需要同时读取以及写入,那么我们就不能使用通配符了。

四、 类型擦除

泛型是 Java 1.5 版本才引进的概念,在这之前是没有泛型的概念的,但显然,泛型代码能够很好地和之前版本的代码很好地兼容。

这是因为,泛型信息只存在于代码编译阶段,在进入 JVM 之前,与泛型相关的信息会被擦除掉

专业术语叫做 类型擦除

List<String> strList = new ArrayList<String>();
List<Integer> integerList = new ArrayList<Integer>();
        
System.out.println(strList.getClass() == integerList.getClass());
==== output =====
true
=================

打印的结果为 true 是因为 List<String>List<Integer>在 jvm 中的 Class 都是 List.class。

泛型信息被擦除了。

/**
 * 类型擦除 相关类
 *
 * @author Richard_yyf
 * @version 1.0 2019/8/29
 */
public class EraseHolder<T> {

    T data;

    public EraseHolder(T data) {
        this.data = data;
    }

    public static void main(String[] args) {
        EraseHolder<String> holder = new EraseHolder<>("hello");
        Class clazz = holder.getClass();
        System.out.println("erasure class is:" + clazz.getName());

        Field[] fs = clazz.getDeclaredFields();
        for ( Field f:fs) {
            // 那我们可不可以说,泛型类被类型擦除后,相应的类型就被替换成 Object 类型呢?
            System.out.println("Field name "+f.getName()+" type:"+f.getType().getName());
        }

        EraseHolder2<String> holder2 = new EraseHolder2<>("hello");
        clazz = holder2.getClass();
        fs = clazz.getDeclaredFields();
        for ( Field f:fs) {
            System.out.println("Field name "+f.getName()+" type:"+f.getType().getName());
        }
    }

    static class EraseHolder2<T extends String> {
        T data;

        public EraseHolder2(T data) {
            this.data = data;
        }
    }
}

局限性

利用类型擦除的原理,用反射的手段就绕过了正常开发中编译器不允许的操作限制。

public class EraseReflectDemo {

    public static void main(String[] args) {
        List<Integer> list = new ArrayList<>();
        list.add(23);
        // can't add here
        // 因为泛型的限制 boolean add(E e);
        list.add("123"); // error

        // 利用反射可以绕过编译器去调用add方法
        // 又因为类型擦除时 boolean add(E e); 等同于 boolean add(Object e);

        try {
            Method method = list.getClass().getDeclaredMethod("add", Object.class);

            method.invoke(list, "test");
            method.invoke(list, 42.9f);
        } catch (Exception e) {
            e.printStackTrace();
        }

        for (Object o : list) {
            System.out.println(o);
        }


    }
}

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

推荐阅读更多精彩内容

  • “泛型”这个术语的意思是:"适用于许多许多的类型”。如何做到这一点呢,正是通过解耦类或方法与所使用的类型之间的约束...
    王侦阅读 1,129评论 0 0
  • 参数类型的好处 在 Java 引入泛型之前,泛型程序设计是用继承实现的。ArrayList 类只维护一个 Obje...
    杰哥长得帅阅读 873评论 0 3
  • 1.泛型的由来 一般的类和方法,只能使用具体的类型,要么是基本数据类型,要么是自定义的类型,如果要编写可以适用于多...
    BigDreamMaker阅读 549评论 0 1
  • ArrayList就是个泛型类,我们通过设定不同的类型,可以往集合里面存储不同类型的数据类型(而且只能存储设定的数...
    dinel阅读 484评论 0 2
  • 1、基本应用 Java泛型可以用在类、接口和方法上。基本使用请参考《on Java 8》. 2、类型擦除 ​ ...
    流_心阅读 312评论 0 0