Java泛型总结

Java泛型总结

Table of Contents

一.概述

1.什么是Java泛型

泛型,即“参数化类型”。就是将类型由原来具体的类型, 比如Food food 这个Food类型改为参数化, 由外部调用的时候来指定。

2.Java泛型的意义

使用泛型的意义在于:

  • a.封装及代码复用, 适用于多种数据类型执行相同的代码(比如Request<T>, Response<T>), 这样就可以通过对外开放相同的接口来完成对一组类的操作
  • b.在编译时期进行类型安全检查 (比如ArrayList<T>)
  • c.控制数据安全访问, 即PECS法则。

二.泛型类

泛型类型用于类的定义中,被称为泛型类。


// 此处T可以随便写为任意标识,常见的如T、E、K、V等形式的参数常用于表示泛型
class GenericClass<T> {
    T data;

    public GenericClass() {

    }

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

    public void setData(T data) {
        this.data = data;
    }

    public T getData() {
        return data;
    }
}
// 在实例化泛型类时,传入的实参类型(说的是右侧的<String>)需与泛型的类型参数类型(说的是左侧的<String>)相同。
// 右侧的可以推断出来, 因此可以省略。但是要写的话,一定要相同。
GenericClass<String> s = new GenericClass<String>("123");

// 在使用泛型的时候如果传入泛型实参,则会根据传入的泛型实参做相应的限制,此时泛型才会起到本应起到的限制作用。
// 如果不传入泛型类型实参的话,在泛型类中使用泛型的方法或成员变量定义的类型可以为任何的类型。
GenericClass generic = new GenericClass("111111");
GenericClass generic1 = new GenericClass(4444);
GenericClass generic2 = new GenericClass(55.55);
GenericClass generic3 = new GenericClass(false);

三.泛型接口

泛型类型用于接口的定义中,被称为泛型类。

interface Generator<T> {
    public T next();
}

泛型接口的实现方式有两种:

// 实现泛型接口的方式1
// 接口中未传入泛型实参,需将泛型的声明也一起加到类中, 也就是在类的声明中写上<T>
class FruitGenerator<T> implements Generator<T> {

    @Override
    public T next() {
        return null;
    }
}

// 实现泛型接口的方式2
// 在接口中指定泛型实参, 这样类中的泛型就规定好了, 所有使用泛型的地方都要替换成传入的实参类型, 这里是String
class FoodGenerator implements Generator<String> {

    @Override
    public String next() {
        return null;
    }
}

四.泛型方法

泛型类,是在实例化类的时候指明泛型的具体类型;泛型方法,是在调用方法的时候指明泛型的具体类型。

1.泛型方法基本介绍

/**
 * 泛型方法的基本介绍
 * @param tClass 传入的泛型实参
 * @return T 返回值为T类型
 * 说明:
 *     1)public 与 返回值中间<T>非常重要,可以理解为声明此方法为泛型方法。
 *     2)只有声明了<T>的方法才是泛型方法,泛型类中的使用了泛型的成员方法并不是泛型方法。
 *     3)<T>表明该方法将使用泛型类型T,此时才可以在方法中使用泛型类型T。
 *     4)与泛型类的定义一样,此处T可以随便写为任意标识,常见的如T、E、K、V等形式的参数常用于表示泛型。
 */
public <T> T genericMethod(Class<T> tClass)throws InstantiationException,
  IllegalAccessException{
        T instance = tClass.newInstance();
        return instance;
}
Object obj = genericMethod(Class.forName("com.test.test"));

2.泛型方法的说明

class GenericMethodTest {
    // 这个类是个泛型类,在上面已经介绍过
    public class Generic<T> {
        private T key;

        public Generic(T key) {
            this.key = key;
        }

        // 这个方法虽然使用了泛型,但是这并不是一个泛型方法。
        // 这只是类中一个普通的成员方法,只不过他的返回值是在声明泛型类已经声明过的泛型。
        // 所以在这个方法中才可以继续使用 T 这个泛型。
        public T getKey() {
            return key;
        }

        /**
         * 这个方法显然是有问题的,在编译器会给我们提示这样的错误信息"cannot resolve symbol E"
         * 因为在类的声明中并未声明泛型E,所以在使用E做形参和返回值类型时,编译器会无法识别。
         *
         *      public E setKey(E key){
         *             this.key = keu
         *      }
         */
    }

    /**
     * 这才是一个真正的泛型方法。
     * 首先在public与返回值之间的<T>必不可少,这表明这是一个泛型方法,并且声明了一个泛型T
     * 这个 T 可以出现在这个泛型方法的任意位置.
     * 泛型的数量也可以为任意多个
     * 如:
     *      public <T,K> K showKeyName(Generic<T> container){
     *          ...
     *      }
     */
    public <T> T showKeyName(Generic<T> container) {
        T test = container.getKey();
        return test;
    }

    /**
     * 这也不是一个泛型方法,这就是一个普通的方法,只是使用了Generic<Number>这个泛型类做形参而已。
     */
    public void showKeyValue1(Generic<Number> obj) {
    }

    /**
     * 这也不是一个泛型方法,这也是一个普通的方法,只不过使用了泛型通配符?
     */
    public void showKeyValue2(Generic<?> obj) {
    }

    /**
     * 这个方法是有问题的,编译器会为我们提示错误信息:"UnKnown class 'E' "
     * 虽然我们声明了<T>,也表明了这是一个可以处理泛型的类型的泛型方法。
     * 但是只声明了泛型类型T,并未声明泛型类型E,因此编译器并不知道该如何处理E这个类型。
     *
     *      public <T> T showKeyName(Generic<E> container){
     *          ...
     *      }
     */
}

3.泛型方法的使用


class GenericFruit {
    static class Fruit {
        @Override
        public String toString() {
            return "fruit";
        }
    }

    static class Apple extends Fruit {
        @Override
        public String toString() {
            return "apple";
        }
    }

    static class Person {
        @Override
        public String toString() {
            return "Person";
        }
    }

    static class GenerateTest<T> {
        public void show_1(T t) {
            System.out.println(t.toString());
        }

        // 这是在泛型类中声明了一个泛型方法,使用泛型E,这种泛型E可以为任意类型。可以类型与T相同,也可以不同。
        // 由于泛型方法在声明的时候会声明泛型<E>,因此即使在泛型类中并未声明泛型,编译器也能够正确识别泛型方法中识别的泛型<E>。
        public <E> void show_3(E t) {
            System.out.println(t.toString());
        }

        //在泛型类中声明了一个泛型方法,使用泛型T,注意这个T是一种全新的类型,可以与泛型类中声明的T不是同一种类型。
        public <T> void show_2(T t) {
            System.out.println(t.toString());
        }
    }

    public static void main(String[] args) {
        Apple apple = new Apple();
        Person person = new Person();

        // 创建一个泛型类,并且指定泛型的实参是Fruit类型
        GenerateTest<Fruit> generateTest = new GenerateTest<Fruit>();

        // 调用类的成员方法(这个show_1()方法是类的成员方法)
        // apple是Fruit的子类,所以这里可以传入Apple类型
        // 单单对泛型T而已, 是可以传入子类对象的(多态)
        generateTest.show_1(apple);


        // 编译器会报错,因为泛型类型实参在创建的时候指定的是Fruit,而传入的实参类是Person
        // generateTest.show_1(person);

        // 调用泛型方法
        generateTest.show_2(apple);
        // 这是完整的写法, 在调用泛型方法的时候指定泛型的类型为Apple
        generateTest.<Apple>show_2(apple);
        // 调用泛型方法
        generateTest.show_2(person);
        // 这是完整的写法, 在调用泛型方法的时候指定泛型的类型为Person
        generateTest.<Person>show_2(person);

        // 这两个也是在调用泛型方法, 与上面的一致
        generateTest.show_3(apple);
        generateTest.show_3(person);

        // 这样调用泛型方法是可以的。指定泛型方法的泛型实参为Fruit类型,然后参入传入Fruit的子类。(多态)
        generateTest.<Fruit>show_3(apple);

        // 总结:

        // a.在指定类的T(也就是泛型实参)的时候, 右侧的Fruit必须和左侧指定的Fruit完全相同,不能写子类。因为:GenericClass<Apple>  与 GenericClass<Fruit>没有继承关系的(具体请看  《 第六节.继承关系 》)
        // 比如 GenericClass<Fruit> n = new GenericClass<Fruit>(new Fruit()); 右侧的Fruit必须和左侧指定的Fruit完全相同
        // 如果 GenericClass<Fruit> n = new GenericClass<Apple>(new Fruit()); 就报错

        // b.在指定类的T(也就是泛型实参)的时候, 传入的方法的参数也一定要是泛型实参的类型,不能是子类。
        // 比如 GenericClass<Fruit> n = new GenericClass<>(new Fruit());
        // 如果 GenericClass<Fruit> n = new GenericClass<>(new Apple()); 是报错的

        // c.但是在调用的时候, 是可以传入 T 的 子类的。
        // 比如
        // OK
        GenericClass<Number> g1 = new GenericClass<Number>(1);
        // NOT OK
        GenericClass<Number> g2 = new GenericClass<Integer>(2);
        // 也 NOT OK
        GenericClass<Fruit> gApple = new GenericClass<Fruit>(new Apple());

        // 但是在调用方法的时候, 如果参数是类型是T, 那可以传入T的子类的
        gApple.setData(new Apple());
    }
}

五.泛型使用过程中的相关限制

  1. 泛型的类型参数只能是类类型(包括自定义类),不能是基本类型
    <br />
  2. 泛型类中的泛型T不能直接实例化, 比如T t = new T(), 这样是不行的。
    <br />
  3. 泛型变量不能作为类的静态变量存在, 只能作为类的成员变量。 比如: private static T t; 这样是不行的。
    3.1. 因此,静态方法是无法访问用泛型声明的成员变量的。
    3.2. 那么,想在静态方法中引入泛型,就必须使用静态的泛型方法, 也就是 staitc <T> void show(T t) {} 这样的形式。
    <br />
  4. 无法使用instanceof关键字判断泛型类的类型, 无法使用“==”判断两个泛型类的实例
        List<String> stringArrayList = new ArrayList<String>();
        List<Integer> integerArrayList = new ArrayList<Integer>();

        /**
         * 无法使用instanceof关键字判断泛型类的类型
         * Illegal generic type for instanceof
         *
         * if(stringArrayList instanceof ArrayList<String>){
         *      return;
         * }
         */

        /**
         * 无法使用“==”判断两个泛型类的实例
         * Operator '==' cannot be applied to this two instance
         *
         * if (stringArrayList == integerArrayList) {
         *      return;
         * }
         */
  1. 泛型类的原生类型与所传递的泛型无关,无论传递什么类型,原生类是一样的。Java中的泛型,只在编译阶段有效。在编译过程中,正确检验泛型结果后,会将泛型的相关信息擦出,并且在对象进入和离开方法的边界处添加类型检查和类型转换的方法。也就是说,泛型信息不会进入到运行时阶段。
        List<String> stringArrayList = new ArrayList<String>();
        List<Integer> integerArrayList = new ArrayList<Integer>();
        List<Number> numberArrayList = new ArrayList<Number>();

        Class classStringArrayList = stringArrayList.getClass();
        Class classIntegerArrayList = integerArrayList.getClass();
        Class classNumberArrayList = numberArrayList.getClass();

        if(classStringArrayList == classIntegerArrayList){
            Log.d("泛型测试","类型相同");
        }
        if(classNumberArrayList == classIntegerArrayList){
            Log.d("泛型测试","类型相同");
        }

六.继承关系

  1. IntegerNumber的子类, 但是ArrayList<Integer>不能被看作为ArrayList<Number>的子类。二者没有继承关系。由此可以看出:同一种泛型可以对应多个版本(因为参数类型是不确定的),不同版本的泛型类实例是不兼容的。

    因此,需要传入ArrayList<Number>作为参数的地方, 不可以传入ArrayList<Integer>

  2. 泛型类可以继承其它泛型类,例如: public class ArrayList<E> extends AbstractList<E>。

    因此,需要传入List<E> 的地方, 可以传入 ArrayList<E>, E必须是相同的

示例:

public class GenericInherit<T> {
    private T data1;
    private T data2;

    public T getData1() {
        return data1;
    }

    public void setData1(T data1) {
        this.data1 = data1;
    }

    public T getData2() {
        return data2;
    }

    public void setData2(T data2) {
        this.data2 = data2;
    }

    public static <V> void setData2(GenericInherit<Father> data2) {

    }

    public static void main(String[] args) {
        //Son 继承自 Father
        Father father = new Father();
        Son son = new Son();
        GenericInherit<Father> fatherGenericInherit = new GenericInherit<>();
        GenericInherit<Son> sonGenericInherit = new GenericInherit<>();
        SubGenericInherit<Father> fatherSubGenericInherit = new SubGenericInherit<>();
        SubGenericInherit<Son> sonSubGenericInherit = new SubGenericInherit<>();

        /**
         * 对于传递的泛型类型是继承关系的泛型类之间是没有继承关系的
         * GenericInherit<Father> 与GenericInherit<Son> 没有继承关系
         * Incompatible types.
         */
        father = new Son();
        //fatherGenericInherit = new GenericInherit<Son>();

        /**
         * 泛型类可以继承其它泛型类,例如: public class ArrayList<E> extends AbstractList<E>
         */
        fatherGenericInherit = new SubGenericInherit<Father>();

        /**
         * 泛型类的继承关系在使用中同样会受到泛型类型的影响
         * setData2()方法的参数是: GenericInherit<Father>, 可以传入它的子类 SubGenericInherit<Father>
         * 但是 GenericInherit<Son> 、SubGenericInherit<Son> 与 GenericInherit<Father> 都没有任何继承关系的 
         */
        setData2(fatherGenericInherit);
        //setData2(sonGenericInherit);
        setData2(fatherSubGenericInherit);
        //setData2(sonSubGenericInherit);

    }

    private static class SubGenericInherit<T> extends GenericInherit<T> {

    }

八.泛型的上下边界和通配符 ?

public class GenericByWildcard {
    private static void print(GenericClass<Fruit> fruitGenericClass) {
        System.out.println(fruitGenericClass.getData().getColor());
    }

    private static void use() {
        GenericClass<Fruit> fruitGenericClass = new GenericClass<>();
        print(fruitGenericClass);
        GenericClass<Orange> orangeGenericClass = new GenericClass<>();
        //类型不匹配,可以使用<? extends Parent> 来解决
        //print(orangeGenericClass);因为GenericClass<Orange> 和 GenericClass<Fruit>没有继承关系
    }

    /**
     * <? extends Parent> 指定了泛型类型的上届
     */
    private static void printExtends(GenericClass<? extends Fruit> genericClass) {
        System.out.println(genericClass.getData().getColor());
    }

    public static void useExtend() {
        GenericClass<Fruit> fruitGenericClass = new GenericClass<>();
        printExtends(fruitGenericClass);
        GenericClass<Orange> orangeGenericClass = new GenericClass<>();
        printExtends(orangeGenericClass);

        GenericClass<Food> foodGenericClass = new GenericClass<>();
        //Food是Fruit的父类,超过了泛型上届范围,类型不匹配
        //printExtends(foodGenericClass);

        //表示GenericClass的类型参数的上届是Fruit
        GenericClass<? extends Fruit> extendFruitGenericClass = new GenericClass<>();
        Apple apple = new Apple();
        Fruit fruit = new Fruit();
        /*
         * 道理很简单,? extends X  表示类型的上界,类型参数是X的子类,那么可以肯定的说,
         * get方法返回的一定是个X(不管是X或者X的子类)编译器是可以确定知道的。
         * 但是set方法只知道传入的是个X,至于具体是X的那个子类,不知道。
         * 总结:主要用于安全地访问数据,可以访问X及其子类型,并且不能写入非null的数据。
         */
        //extendFruitGenericClass.setData(apple);
        //extendFruitGenericClass.setData(fruit);

        fruit = extendFruitGenericClass.getData();

    }

    /**
     * <? super Child> 指定了泛型类型的下届
     */
    public static void printSuper(GenericClass<? super Apple> genericClass) {
        System.out.println(genericClass.getData());
    }

    public static void useSuper() {
        GenericClass<Food> foodGenericClass = new GenericClass<>();
        printSuper(foodGenericClass);

        GenericClass<Fruit> fruitGenericClass = new GenericClass<>();
        printSuper(fruitGenericClass);

        GenericClass<Apple> appleGenericClass = new GenericClass<>();
        printSuper(appleGenericClass);

        GenericClass<HongFuShiApple> hongFuShiAppleGenericClass = new GenericClass<>();
        // HongFuShiApple 是Apple的子类,达不到泛型下届,类型不匹配
        //printSuper(hongFuShiAppleGenericClass);

        GenericClass<Orange> orangeGenericClass = new GenericClass<>();
        // Orange和Apple是兄弟关系,没有继承关系,类型不匹配
        //printSuper(orangeGenericClass);

        //表示GenericClass的类型参数的下界是Apple
        GenericClass<? super Apple> supperAppleGenericClass = new GenericClass<>();
        supperAppleGenericClass.setData(new Apple());
        supperAppleGenericClass.setData(new HongFuShiApple());
        /*
         * ? super  X  表示类型的下界,类型参数是X的超类(包括X本身),
         * 那么可以肯定的说,get方法返回的一定是个X的超类,那么到底是哪个超类?不知道,
         * 但是可以肯定的说,Object一定是它的超类,所以get方法返回Object。
         * 编译器是可以确定知道的。对于set方法来说,编译器不知道它需要的确切类型,但是X和X的子类可以安全的转型为X。
         * 总结:主要用于安全地写入数据,可以写入X及其子类型。
         */
        //supperAppleGenericClass.setData(new Fruit());

        //get方法只会返回一个Object类型的值。
        Object data = supperAppleGenericClass.getData();
    }

    /**
     * <?> 指定了没有限定的通配符
     */
    public static void printNonLimit(GenericClass<?> genericClass) {
        System.out.println(genericClass.getData());
    }

    public static void useNonLimit() {
        GenericClass<Food> foodGenericClass = new GenericClass<>();
        printNonLimit(foodGenericClass);
        GenericClass<Fruit> fruitGenericClass = new GenericClass<>();
        printNonLimit(fruitGenericClass);
        GenericClass<Apple> appleGenericClass = new GenericClass<>();
        printNonLimit(appleGenericClass);

        GenericClass<?> genericClass = new GenericClass<>();
        //setData 方法不能被调用, 甚至不能用 Object 调用;
        //genericClass.setData(foodGenericClass);
        //genericClass.setData(new Object());
        //返回值只能赋给 Object
        Object object = genericClass.getData();

    }

}

PECS法则

PECS指“Producer Extends,Consumer Super”。换句话说,如果参数化类型表示一个生产者,就使用<? extends T>;如果它表示一个消费者,就使用<? super T>。

  1. 如果你想从一个数据类型里获取数据,那么它就是生产者,我们要从里面get()数据,这样的使用 ? extends 通配符2. 如果你想把对象写入一个数据结构里,那么他就是消费者,我们要把数据丢入这个数据结构(set()), 这时候使用 ? super 通配符
  2. 如果你既想存,又想取,那就别用通配符。

PESC法则请参考:

九.获取泛型的参数类型

参考: TypeToken

十.Gson中的TypeToken

参考: TypeToken

总结的内容,大部分参考并摘录了:
https://www.jianshu.com/p/986f732ed2f1
https://blog.csdn.net/s10461/article/details/53941091

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