Java泛型

基本概念

  • 通常情况下集合中可以存放不同类型的对象,是因为将所有对象都看做Object类型放入的,因此从集合中取出元素时也是Object类型,为了表达该元素真实的数据类型,则需要强制类型转换,而强制类型转换可能会引发类型转换异常。

例子:

public class ListTest {
    public static void main(String[] args) {
        List list = new LinkedList();
        list.add(0, 1);
        list.add(1, true);
        list.add(2, "String");
        String s = (String)list.get(2);
        System.out.println(s);
    }
}
  • 为了避免上述错误的发生,从Java5开始增加泛型机制,也就是在集合名称的右侧使用<数据类型>的方式来明确要求该集合中可以存放的元素类型,若放入其它类型的元素则编译报错。
  • 泛型只在编译时期有效,在运行时期不区分是什么类型。
public class ListGenericityTest {
    public static void main(String[] args) {
        // 1.准备一个支持泛型机制的List集合,明确要求集合中的元素是String类型
        List<String> lt1 = new LinkedList<>();
        // 2.向集合中添加元素并打印
        lt1.add("one");
        System.out.println("lt1 = " + lt1); // one
        // lt1.add(2); Error
        // 3.获取集合中的元素并打印
        String s = lt1.get(0);
        System.out.println("获取到的元素是:" + s); // one
        System.out.println("============================================================");
        // 4.准备一个支持Integer类型的List集合
        List<Integer> lt2 = new LinkedList<>();
        lt2.add(1);
        lt2.add(2);
        // lt2.add("3"); Error
        System.out.println("lt2 = " + lt2); // [1, 2]
        Integer integer = lt2.get(0);
        System.out.println("获取到的元素是:" + integer); // 1
        System.out.println("============================================================");
        // Java7开始的新特性:菱形特性,也就是后面<>中的数据类型可以省略
        List<Double> lt3 = new LinkedList<>();
        // 笔试考点
        // 视图将lt1的数值赋值给lt3,也就是覆盖lt3中原来的数据,结果会编译报错
        // 报错原因是集合中支持的类型不同
        // lt3 = lt1; Error
    }
}

结果为:

lt1 = [one]
获取到的元素是:one
============================================================
lt2 = [1, 2]
获取到的元素是:1
============================================================

底层原理

  • 泛型的本质就是参数化类型,也就是让数据类型作为参数传递,其中E相当于形式参数负责占位,而使用集合时<>中的数据类型相当于实际参数,用于给形式参数E进行初始化,从而使得集合中所有的E被实际参数替换,由于实际参数可以传递各种各样广泛的数据类型,因此得名为泛型。
  • 如:
其中i叫做形式参数,负责占位 其中E叫做形式参数,负责占位
int i = 10;int i = 20; E = String; E = Integer;
public static void show(int i) {...} public interface List<E> {...}
其中10叫做实际参数,负责给形式参数初始化 其中String叫做实际参数
show(10); List<String> lt1 = ...;
show(20); List<String> lt2 = ...;

自定义泛型接口

  • 泛型接口和普通接口的区别就是后面添加了类型参数列表,可以有多个类型参数,如:<E, T, .. >等。

自定义泛型类

  • 泛型类和普通类的区别就是类名后面添加了类型参数列表,可以有多个类型参数,如:<E, T, .. >等。
  • 实例化泛型类时应该指定具体的数据类型,并且是引用数据类型而不是基本数据类型。
  • 父类有泛型,子类可以选择保留泛型也可以选择指定泛型类型。
  • 子类必须是“富二代”,子类除了指定或保留父类的泛型,还可以增加自己的泛型。

例子:
先定义泛型类:

/**
 * 自定义泛型类Person,其中T相当于形式参数负责占位,具体数值由实参决定
 * @param <T> 看做是一种名字为T的数据类型即可
 */
public class Person<T> {
    private String name;
    private int age;
    private T gender;
    
    public Person() {
    }
    
    public Person(String name, int age, T gender) {
        this.name = name;
        this.age = age;
        this.gender = gender;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public T getGender() {
        return gender;
    }
    public void setGender(T gender) {
        this.gender = gender;
    }
    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", gender=" + gender +
                '}';
    }
}

测试类:

public class PersonTest {
    public static void main(String[] args) {
        // 1.声明Person类型的引用指向Person类型的对象
        Person p1 = new Person("zhangfei", 30, "男");
        // 2.打印对象的特征
        System.out.println(p1);

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

        // 3.在创建对象的同时指定数据类型,用于给T进行初始化
        Person<String> p2 = new Person<>();
        p2.setGender("女");
        System.out.println(p2);

        // 4.使用Boolean了类型作为性别的类型
        Person<Boolean> p3 = new Person<>();
        p3.setGender(true);
        System.out.println(p3);
    }
}

结果为:

Person{name='zhangfei', age=30, gender=男}
============================================================
Person{name='null', age=0, gender=女}
Person{name='null', age=0, gender=true}

泛型类被继承时的处理方式
子类继承是有四种继承方式的
定义泛型类的子类:

public class SubPerson extends Person{// 不保留泛型而且没有指定类型,此时Person类中的T默认为是Object类型 擦除
}

测试会发现子类调用set方法时需要传入的参数为Object类型
public class SubPerson extends Person<String>{// 不保留类型但是指定了泛型的类型,此时Person类型中的T被指定为String类型
}

此时发现子类调用set方法时需要传入的参数为String类型
public class SubPerson<T> extends Person<T> {// 保留父类的泛型,可以在构造对象时来指定T的类型
}
public class SubPerson<T, T1> extends Person<T> {// 保留父类的泛型,同时在子类中增加新的泛型
}

自定义泛型方法

  • 泛型方法就是我们输入参数的时候,输入的是泛型参数,而不是具体的参数。我们在调用这个泛型 方法的时需要对泛型参数进行实例化。
  • 泛型方法的格式:
    [访问权限] <泛型> 返回值类型 方法名([泛型标识 参数名称]) { 方法体; }
  • 在静态方法中使用泛型参数的时候,需要我们把静态方法定义为泛型方法。

像这种不是泛型方法:

public T getGender() {
    return gender;
}

像这种才是泛型方法(在Person类中新定义的方法):

// 自定义方法实现将参数指定数组中的所有元素打印出来
public <T1> void printArray(T1[] arr) {
    for (T1 t : arr) {
        System.out.println("t = " + t);
    }
}

然后调用泛型方法:

// 5.调用泛型方法进行测试
Integer[] arr = {11, 22, 33};
Person.printArray(arr);

结果为:

t = 11
t = 22
t = 33

一种特殊情况:

// 不是泛型方法,该方法不能使用static关键字修饰,因为该方法中的T需要在new对象时才能明确类型
public /*static*/ T getGender() {
    return gender;
}

泛型在继承上的体现

  • 如果B是A的一个子类或子接口,而G是具有泛型声明的类或接口,则G<B>并不是G<A>的子类型!比如:String是Object的子类,但是List<String>并不是List<Object>的子类。

通配符的使用

  • 有时候我们希望传入的类型在一个指定的范围内,此时就可以使用泛型通配符了。
  • 如:之前传入的类型要求为Integer类型,但是后来业务需要Integer的父类Number类也可以传入。
  • 泛型中有三种通配符形式:
    <?> 无限制通配符:表示我们可以传入任意类型的参数。
    <? extends E> 表示类型的上界是E,只能是E或者是E的子类。
    <? super E> 表示类型的下界是E,只能是E或者是E的父类。

例子:
先定义父类

public class Animal {
}

再定义子类,继承父类

public class Dog extends Animal{
}

测试类

public class GenericTest {
    public static void main(String[] args) {
        // 1.声明两个List类型的集合进行测试
        List<Animal> lt1 = new LinkedList<>();
        List<Dog> lt2 = new LinkedList<>();
        // 试图将lt2的数值赋值给lt1,也就是发生List<Dog>类型向List<Animal>类型的转换
        //lt1 = lt2;  Error: 类型之间不具备父子类关系

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

        // 2.使用通配符作为泛型类型的公共父类
        List<?> lt3 = new LinkedList<>();
        // 没报错,说明可以发生List<Animal>类型到List<?>类型的转换
        lt3 = lt1;
        // 可以发生List<Dog>类型到List<?>类型的转换
        lt3 = lt2;

        // 向公共父类中添加元素和获取元素
        // lt3.add(new Animal()); Error: 不能存放Animal类型的对象
        // lt3.add(new Dog());    Error: 不能存放Dog类型的对象,不支持元素的添加操作,因为?可以是任意类型

        // 是可以的,支持元素的获取操作,全部当做Object类型来处理
        Object o = lt3.get(0);

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

        // 3.使用有限制的通配符进行使用
        List<? extends Animal> lt4 = new LinkedList<>();
        // 不支持元素的添加操作,因为?可以是Animal以及任意Animal的子类类型
        // 如果?为Dog类型,那么添加new Animal()对象就是错误的
        //lt4.add(new Animal());
        //lt4.add(new Dog());
        //lt4.add(new Object());
        // 获取元素是可以的,因为取出来的只能是Animal或者Animal的子类
        Animal animal = lt4.get(0);

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

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

推荐阅读更多精彩内容