泛型与集合

泛型与集合

直接开冲

泛型

本篇博客主要参考码出高效-Java开发手册

泛型的基本概念

泛型的定义和本质

  1. 泛型指的是可以将类型作为参数进行传递,其本质上就是类型参数化,泛型只会存在于编译期,在运行期会进行参数擦除。关于泛型的擦除,如果泛型没有设置类型上限,则泛型会转化为Object类型,如果设置上限则会转化为设置的上限。最后再进行类型的强转,转化为我们需要的类型。
  1. 除了基本类型之外,其他的都可以都可以作为泛型。泛型可以定义在类、接口、方法中,编译器通过尖括号和尖括号中的字母来解析泛型,有些字母是约定俗成的:
    • E代表Element,表示集合中的元素。
    • T代表the Type of object,表示某个类
    • K代表key,V代表value,用于键值对元素
      下面代码选自码出高效:
    public class GeniecTestDemo<T> {
    /**
     * @param type0
     * @param type1
     * @param <String>
     * @param <T>
     * @param <Alibaba>
     * @return
     */
    public static <String, T, Alibaba> String getString(String type0, Alibaba type1) {
        return type0;
    }
    
    public static void main(String[] args) {
        Integer integer = Integer.valueOf("123");
        Long aLong = Long.valueOf("232324");
        //返回的值也是第一个传入的类型的泛型
        Integer string = getString(integer, aLong);
    
    }
    }
    

针对本段代码的说明:

  • 此处的String不是java.lang.String中的String,而是自己定义的泛型,和泛型T、Alibaba效果一样
  • 类名后的泛型T和方法中的泛型T互相不影响,相互独立
  • 尖括号的位置在类名之后或者在方法返回值之前
  • 泛型在定义处只有执行Object方法的能力,即在get方法内部无法使用type0.intValue和type1.longValue方法完成

泛型的优势:

- 避免类型的强制转换
- 在编译期内进行更强的类型检查
- 增强代码的复用

泛型的使用

泛型定义位置

  • 泛型定义的位置:

    1. 泛型定义在方法上:在调用方法时,可以使用任何类型的参数;泛型定义必须放在返回值前,修饰词后。
    2. 泛型定义在类上,创建该类对象时,明确这个对象使用的参数类型。
  • 静态方法能否使用泛型

    1. 静态方法不能使用类上定义的泛型,因为泛型是需要通过对象来明确的,静态方法不需要对象,所以无法用该泛型。
    2. 如果静态方法要使用泛型,只能将泛型定义在方法上。

泛型的使用细节

<T>、<?>、Object与List、List<?>、List<Object>
解决这个问题首先弄清楚这三位是什么,以及在JDK源码中的应用,第一个是泛型类型参数T,第二个是泛型通配符,第三个是Object类型.


经典问题,到底我选哪个
类型参数和Object类的区别

其实在泛型没推出来之前,在集合容器中使用Object类很频繁,泛型推出来之后给我们带来的是使用的便利,具体的优势如下:

  1. 类型安全:放入什么、取出什么,不用担心抛出ClassCastException
  2. 提升可读性:编码阶段显示知道泛型集合、泛型方法等处理对象的类型是什么
  3. 提高代码的重用性
    看起来有点眼熟是不是?这不就是为什么要是用泛型吗?是的。
泛型通配符
泛型通配符含义

泛型通配符含义:?代表的是未知类型,所以涉及到?的操作,一定与具体类型无关。

泛型通配符的形式
  • <?> 无限通配符 一般搭配容器使用,只有读的能力,没有写的能力
  • <? extends T> 定义了泛型上限,限定了泛型的类型只能是T或者是T的子类,只有读的能力,没有写的能力
  • <? super T> 定义了泛型的下限,泛型的参数只能是T或者是T的父类,有读取的能力和部分写的能力
泛型通配符?和泛型参数T

关于这个问题,这个老师讲的比较透彻,本文也是吸取了部分他的内容。

https://blog.csdn.net/sinat_32023305/article/details/83215751

选什么,老师把好处都说了


全都要的爪巴
  1. 首先是两者可以互换的地方
  • List<?> mlist、List<T> mlistT
  • List<? extends A> mlist、List<T extends A> mlistT
    在这两个地方中通配符?和泛型参数T是一样的
  1. 两者的区别
  • List<?>表示不确定的类型,代表范围内任意类型,List<T>表示确定的类型(一旦在创建对象或者方法调用时该类型即是已经确定),代表范围内所有类型
    [图片上传失败...(image-c86b0e-1597131632906)]
    从使用者角度出发,通配符代表的泛型是不确定的,即使在方法中调用或者在初始化对象时将该泛型确定之后,对于该泛型也只能做读取、判空等操作,但是不能增加元素;
    这边需要对读取做两个说明,
  1. 当返回值需要和泛型相关时,就不能使用。
  2. 不能对数据进行增加操作增加的操作,
  • 只有泛型通配符?才支持super关键字,泛型T不支持super关键字,无法对泛型的下限进行限定

  • 通配符中数据不能增加,原因在前文已经说明了。泛型参数T可读取、可修改。泛型参数T在类型实例化的时候,泛型的类型就已经确定。

  1. 两者的使用
  • 通配符的优势在于代码简洁,可读性好,因而对于数据我们如果只需要读取,并且返回值对该泛型没有任何依赖时,选择使用?,否则使用泛型参数T。
    在JDK源码中,这部分代码在集合类Collections中表现的很明显
 public static int indexOfSubList(List<?> source, List<?> target) {
     .....
 }

该方法作用是查找集合的某一子集在该集合中的位置坐标,对数据不需要进行修改、增加的操作,因而选择通配符没有选择泛型参数

  • 在日常的使用过程中,两者往往是配合使用,比如日常使用频率比较高的Collections的排序
   public static <T> void sort(List<T> list, Comparator<? super T> c) 
List<?> List List<Object>

这个问题出自码出高效,书中也是做了解答。


泛型未推出时定义List的方式

完整代码图片

对于非泛型集合List,可以赋值给任意泛型集合,所以图中的几个集合的定义没有问题。

  1. 首先是List和List<Object>
    看起来没有区别不是吗?


    你说的不对

    两者都可以添加任意元素(Java中Object类是所有类的父类),但是在赋值时就会出现问题了。
    请看下面的代码
    这段代码有两个问题

 public static void main(String[] args) {
        //List和List<Object>的区别
        List<Object> objects = new ArrayList<>();
        List mLists = new ArrayList();
        List<Long> longLists=objects;
        List<Long> longLists1=mLists;

        //数组和集合的协变性,
        ArrayList<Integer> integers = new ArrayList<>();
        integers.add(1234);
        List<Object> Objects1=integers;

        Integer[] integers1 = {1};
        Object [] mObjects1=integers1;

    }
image

这段代码有两个问题:

  1. 将Object类型的集合赋值给Long型的很显然是不可以的,因为List<Object>中对集合的泛型类型作了限制,只能接受Object类型的参数,将Object类型的集合赋值给Integer类型的集合显然是不符合规范
  2. 按照数组可以协变,那么集合的泛型是否可以协变呢,关于协变,可以参照这个大佬的博客

https://blog.csdn.net/magi1201/article/details/45127487

答案是不可以。
数组的协变性(covariant)是指如果类Base是类Sub的基类,那么Base[]就是Sub[]的基类。而泛型是不可变的,List<Base>不会是List<Sub>的基类,更不会是它的子类。
对照代码部分就是integer类型的集合是不可以赋值给Object类型的,在编译阶段就会报错。

  1. 关于List<?>则相对简单,List<?> 不支持添加任何元素,可以remove和clear元素,List<?> 作为参数接受外部的集合,或者返回不知道具体类型的集合
List<? extends T> 和 List<? super T>区别

List<T>只能放置一种类型,为放置解决多种受泛型约束的类型,出现了List<?extends T>和List<? super T>这两种

  1. 首先我们看一下结论,List<? extends T>的特点如下:
  • 可赋值给T及T的子类集合
  • 除了null之外,任何元素不得添加进<? extends T>中
    综合以上特点,我们可知List<? extends T>适合消费元素为主的场景,取出元素反而能加上限
  1. List<? super T>的特点:
  • 可赋值给T及T的父类
  • 可添加T及T的父类,下界为T
  • 取出功能受限,只能取出Object类型的对象
    综合以上特点,List<? super T>适合生产元素为主的场景,取出元素会发生类型擦除。
    下面看示例代码:


    image

    这段代码可以看到有明显的编译错误:

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