JAVA泛型和通配符,再也不用每次百度了

【概述】


泛型机制在项目中一直都在使用,比如在集合中ArrayList<String, String>, Map<String,String>等,不仅如此,很多源码中都用到了泛型机制,所以深入学习了解泛型相关机制对于源码阅读以及自己代码编写有很大的帮助。但是里面很多的机制和特性一直没有明白,特别是通配符这块,对于通配符上界、下界每次用每次百度,经常忘记,这次我就做一个总结,加深自己的理解。

【泛型介绍和使用】

泛型在类定义时不会设置类中的属性或方法参数的具体类型,而是在类使用时(创建对象)再进行类型的定义。会在编译期检查类型是否错误, 保证程序的可读性和安全性。

泛型定义根据实际情况可以分为泛型类和泛型方法:

>>>泛型类

public class Point<T, U> {

    private T pointX;

    private U pintY;

    public Point(T pointX, U pintY) {

        this.pointX = pointX;

        this.pintY = pintY;

    }

    public void showPoint() {

        System.out.println(pointX);

        System.out.println(pintY);

    }

}

●类中引入类型变量,类型变量指的T, U这些,用尖括号<>括起来, 跟在类名后面。

●多个类型变量可以用逗号分隔。

●在类中的方法和返回值等地方可以使用类型变量。

●类型变量采用大写形式,要求简短,一般E表示集合的元素类型,K和V表示key和value等。

泛型类使用:Point<Integer, Double>

>>>泛型方法

public class FxMethod {

    public static <T> T getMiddleNumber(T ... numbers) {

        return null;

    }

    public <T, U> void showNumber(T t, U u) {

        System.out.println("t = " + t);

        System.out.println("u = " + u);;

    }

}

●方法中引入类型变量,在返回类型前添加<>, 中间放置类型变量,多个类型变量用逗号分隔。

●在方法的参数和返回值等位置可以使用类型变量。

●泛型方法使用:Integer result = FxMethod.getMiddleNumber(2, 3)或者Integer result = FxMethod.<Integer>getMiddleNumber(2, 3)。

>>>类型变量的限定

前面讲解了泛型一般定义的两种方式,其中的类型变量没有任何限定, 这样在导致一方面在定义泛型的时候无法使用一些API, 需要强转,另一方面在使用的时候也容易出错,那么如何给类型变量添加限定呢?

●只有通过extends关键字限定,不能通过super关键字。

●加了限定以后,就可以直接使用限定类相关的API。

●多个限定之间用&符号,比如T extends Number & Comparable。

●使用泛型时,只能传入相应限定的类,比如传入Point<String, String> 就会报编译错误。

【通配符使用】

泛型的引入的确解决了很大问题,那它是完美的吗?

class AnimalWrapper<T extends Animal> {

    private T animal;

    AnimalWrapper(T animal) {

        this.animal = animal;

    }

    public void eat() {

      animal.eat();

    }

}

class Animal {

    private String name;

    public void eat() {

        System.out.println("animal eat -----");

    }

}

class Cat extends Animal {

    @Override

    public void eat() {

        System.out.println(" cat eat -----");

    }

}

定义一个AnimalWrapper,泛型变量中限定为Animal,如果是下面的测试类,会怎么样呢?

会编译报错,因为AnimalWrapper并不是AnimalWrapper的子类,不能直接传入。为了解决个问题,我们引入了通配符,通配符一般是在方法中或者泛型类使用中用到。

AnimalWrapper可以作为AnimalWrapper<?extends Animal>的子类型,这就是利用通配符带来的好处。

●统配符使用在集合或者方法的参数返回值中。

●通配符可以分为无边界通配符、上边界通配符和下边界通配符。

>>>无边界通配符

通配符无边界,可以传入任何类型,没有限制,相当于Object.

基本语法:

<?>

例子:

public static void printList1(List<?> list) {

        for (Object x:list) {

            System.out.println(x);

        }

    }

    public static void main(String[] args) {

        List<Integer> list = new ArrayList<>();

        list.add(1);

        printList1(list);  // ok

        List<String> list2 = new ArrayList<>();

        list2.add("1");

        printList1(list2); // ok


        List<?> list3 = list;

        // get只能用Object接受,

        Object o = list3.get(0);

        list3.add(5);  // compile error

        list3.add(new Object()); // compile error

    }

小结:

●无边界通配符相当于Object,任何类型都可以传入,比如 List<Integer> list, List<String> list2 。

●由于?无法确定是哪种类型,所以只能使用Object类型的变量接收, 比如例子中的:Object o = list3.get(0);

●如果是无边界通配符对应的集合类型,不能添加任何元素。因为无法确定集合存放数据的类型,鬼知道我们要放什么类型才合适啊。

>>>通配符上界

通配符上界,可以限制传入的类型必须是上界这个类或者是这个类的子类。

基本语法:

<? extends 上界>

<? extends Number>//可以传入的实参类型是Number或者Number的子类

例子:

public static void printList1(List<? extends Number> list) {

        for (Object x:list) {

            System.out.println(x);

        }

    }

    public static void main(String[] args) {

        List<Integer> list = new ArrayList<>();

        list.add(1);

        printList1(list);  // ok

        List<Double> list1 = new ArrayList<>();

        list1.add(1.0D);

        printList1(list1);  // ok

        List<String> list2 = new ArrayList<>();

        list2.add("1");

        printList1(list2); // compile error

        List<? extends Number> list3 = list;

        // get能用上界

        Number o = list3.get(0);

        // 不能add

        list3.add(5);  // compile error

        list3.add(new Object()); // compile error

    }

小结:

●通配符上界? extends A, 表明所有的是A的类或者子类型可以传入,比如本例中的Integer和Double都是Number的子类,String不是。

●通配符上界? extends A,确定了类型是A或者是A的子类,那么向集合容器get获取数据,肯定是它的上界类A,因为其他放的类都是A的子类,比如例子中的 Number o = list3.get(0);

●如果向通配符上界集合中添加元素时,会失败。 List, 说明容器可以容纳的是A或者A的子类,但A的子类有很多,不确定放哪个,为了安全性,就直接不让你add,比如例子中的list3.add(5); ,5虽然是Number的子类,依然不能add。

>>>通配符下界

通配符下界,可以限制传入的类型必须是这个类或者是这个类的父类。

基本语法:

<? super 下界>

<? super Integer>//代表 可以传入的实参的类型是Integer或者Integer的父类类型

例子:

public static void printList1(List<? super Integer> list) {

        for (Object x:list) {

            System.out.println(x);

        }

    }

    public static void main(String[] args) {

        List<Integer> list = new ArrayList<>();

        list.add(1);

        printList1(list);  // ok

        List<Double> list1 = new ArrayList<>();

        list1.add(1.0D);

        printList1(list1);  // compile error

        List<String> list2 = new ArrayList<>();

        list2.add("1");

        printList1(list2); // compile error

        List<? super Integer> list3 = list;

        // 不能用下界接收

        Integer o = list3.get(0); // compile error

        // 能add

        list3.add(5);  // ok

        list3.add(new Number(5)); // compile error

    }

●通配符上界? super A, 表明所有的是A的类或者A的父类可以传入。

●通配符上界? super A,确定了类型是A或者是A的父类,那么向集合容器get获取数据,无法确定是A还是A的某个父类,所以不能get,Integer o = list3.get(0); // compile error,比如例子中用Integer接收,万一list3中放的是Object类型,就凉凉了。

●如果向通配符下界集合中添加元素时,只能添加下届类的子类。比如例子中的:list3.add(5), list3的通配符是<? super Integer>,说明该集合存放的是Integer或者Integer的父类,我只要向容器中放Integer和它的子类都是成立的。

【总结】

本文浅谈了下泛型和通配符的使用,是自己理解的总结,希望后面的开发过程中不要再去百度了,如果哪里有问题希望大家指正。

原文链接:https://juejin.cn/post/7140099537703534600

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

推荐阅读更多精彩内容