一起来学习Java的泛型

目录

  • 泛型:
    • 泛型父类和子类:
    • 泛型接口:
    • 泛型方法:
    • 通配符:
      • 举例说明:
  • 总结
    【一起来学习Java的泛型】
    泛型: 什么是泛型?
    泛型是在Java SE 1.5引入的的新特性,本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。这种参数类型可以用在类、接口和方法的创建中,分别称为泛型类、泛型接口、泛型方法。
    简而言之:<>泛型就是用来约束类、方法、属性上的数据类型,比如
    List list = new ArrayList(); List list = new ArrayList();

    new ArrayList这个集合的元素只能添加Integer类型。
    为什么需要泛型?
    Java推出泛型之前,程序员可以构建一个Object类型的集合,该集合能够存储任何的数据类型,而在使用该 集合的时候,需要程序员明确知道每个元素的具体的类型并向下转型,否则容易引发ClassCastException 类转换异常。现在我们通过泛型就能解决这个问题,通过泛型<>我们就能够约束这个集合插入的类型,就不需要再从Object类型转换成子类类型。
    泛型有什么好处?
    • 类型安全,不会插入指定类型以外的数据
    • 消除了强制类型转换
    泛型的类型:
    泛型可以定义在类上、父类上、接口上、子类上、方法上、参数上、属性上, 泛型类型是可以用任意字母来代替你需要传递的数据,一般为了可读性,我们约定一般会写:
    • E -element 代码集合中存放的元素
    • T -Type表示类型Java类
    • K -key 表示键
    • V -V 表示value
    • N -Number 表示数值类型
    • ? 表示不确定的类型
    那么这么多字母有什么用呢?我们来看一段代码:这是一个普通的Student类:
    package Test; public class Student {private String name; private int age; 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; }}

    我们升级一下,把这个类定为String的泛型,意思是这个类只能接收String的类型:
    public class Student {//重复的代码就不重写了,和上面的是一样}

    ok,这样没问题,但是我们想一想,如果直接就把这个类的泛型定义死了,如果后面需要在Student传入Integer类型,我们就得重写一个Student类,这样代码的复用性不高,不符合编程思维,所以我们可以这样:
    public class Student {private T t; public T getT() {return t; }public void setT(T t) {this.t = t; }}

    然后当需要用什么泛型的时候直接传入即可:
    Student stu1 = new Student<>(); Student stu2 = new Student<>(); Student stu3 = new Student<>();

    测试:(接上面的代码)
    public class Test {public static void main(String[] args) {//先使用Integer泛型,传入100Student stu1 = new Student<>(100); Integer in = stu1.getT(); System.out.println(in); }}

    输出:100 (没问题)
    但是如果在定义为Integer下输入String类型,会怎么样?
    一起来学习Java的泛型
    文章图片

    很明显,直接就报错了
    这样就更好的利用了代码的复用性。上面部分的知识点也就是泛型类。

    泛型父类和子类:
    我们先定义一个泛型父类:()
    public class Student {private T t; public T getT() {return t; }public void setT(T t) {this.t = t; }}

    然后定义一个子类,继承该Student父类:
    public class Child extends Student{@Overridepublic T getT(){return super.getT(); }@Overridepublic void setT(T t){super.setT(t); }}

    特别需要注意:
    1.泛型子类的参数一定要和父类的参数类型一致
    2.如果子类没有添加泛型,那么父类的参数类型必须明确
    也就是这样:
    public class Child extends Student{}

    测试:下面代码的解释:在父类Student内定义泛型为String,然后创建子类Child的对象child,因为是继承关系,所有child的泛型也是String,
    public class Test {public static void main(String[] args) {Student child = new Child<>(); //多态child.setT("abc"); String value = https://www.it610.com/article/child.getT(); System.out.println(value); }}

    当子类传入非String类型值时:
    一起来学习Java的泛型
    文章图片


    泛型接口:
    泛型接口其实也很简单,定义如下:
    public interface USB {}

    泛型子类实现泛型接口,除了处理标识父类的泛型标识外,还可以继续扩展泛型:
    public class phone implements USB{//这里的意思是子类除了有USB接口的泛型T类,也可以再扩展一个泛型private T color; //手机颜色private V phoneName; //手机名称public phone(T color,V phoneName) {this.color = color; this.phoneName = phoneName; }public T getColor() {return color; }public void setColor(T color) {this.color = color; }public V getPhoneName() {return phoneName; }public void setPhoneName(V phoneName) {this.phoneName = phoneName; }}

    注意: 泛型接口也和泛型父子类一样,如果子类没有添加泛型参数,那么父类一定要明确的指定类型!
    public class phone implements USB{}

    测试:
    分别把T和V的泛型都定义为String:
    public class Test {public static void main(String[] args) {phone ph = new phone<>("黑色","华为"); String str1 = ph.getColor(); String str2 = ph.getPhoneName(); System.out.println(str1+str2); }}


    泛型方法:
    定义泛型方法:在public和返回值之间定义<>的才是泛型方法:
    public void printType(){}

    然而这个方法的返回值类型就是取决于E的类型,如果E是Integer,那么这个方法的返回值类型就是整数。用法基本和上述的一致,需要什么类型的返回值,在调用的时候去传递泛型类型即可。

    通配符:
    通配符一般是使用 ? 代替具体的类型实参(此处是类型实参,而不是类型形参)。当操作类型时不需要使用类型的具体功能时,只使用Object类中的功能,那么可以用 ? 通配符来表未知类型。例如 List 在逻辑上是List、List 、List等所有List<具体类型实参>的父类。
    有界的类型参数:
    有的时候需要限制那些被允许传递到一个类型参数的类型种类范围,例如一个操作数字的方法可能只希望接受Number或者Number子类的实例。这时就需要为泛型添加上边界,即传入的类型实参必须是指定类型的子类型。要声明一个有界的类型参数,首先列出类型参数的名称,后跟extends或super关键字,最后紧跟它的上界或下界。由此可以知道泛型的上下边界的添加必须与泛型的声明在一起 。
    上限:
    表示该通配符所代表的类型是T类型的子类。 例如往集合中添加元素时,既可以添加T类型对象,又可以添加T的子类型对象。
    下限:
    表示该通配符所代表的类型是T类型的父类,就是只能获取到T类及以上的泛型,任何继承T类的泛型将得不到。

    举例说明: 建一个动物的父类:
    public class Animal {}

    建一个猫类,继承父类:
    public class Cat extends Animal{}

    建一个小猫类,继承猫类:
    public class MiniCat extends Cat{}

    现在的关系是Animal>Cat>MiniCat
    通配符上限的测试类:
    public class test {public static void main(String[] args) {ArrayList animals = new ArrayList<>(); ArrayList cats = new ArrayList<>(); ArrayList minicats = new ArrayList<>(); showAnimals(cats); showAnimals(minicats); //这样会报错,因为在showAnimals方法内的通配符最高继承自CatshowAnimals(animals); }//使用通配符,把上线限制到Cat,意思是Animal这个类不能被获取private static void showAnimals(ArrayList cats) {for (Cat cat:cats) {System.out.println(cat); }}}

    一起来学习Java的泛型
    文章图片

    可以看到使用通配符限制到Cat类后,Animals这个类就已经获取不到了。这就是上限。
    如果把上限设置成Animal的时候:
    一起来学习Java的泛型
    文章图片

    通配符下限的测试类:(其他代码与上面的一致)
    public class test {public static void main(String[] args) {ArrayList animals = new ArrayList<>(); ArrayList cats = new ArrayList<>(); ArrayList minicats = new ArrayList<>(); showAnimals(cats); showAnimals(minicats); //这样会报错,因为在showAnimals方法内的通配符最高继承自CatshowAnimals(animals); }//使用通配符,把上线限制到Cat,意思是Animal这个类不能被获取private static void showAnimals(ArrayList cats) {for (Cat cat:cats) {System.out.println(cat); }}}

    因为通配符设置了下限到Cat类,所以MiniCat是获取不到的:
    一起来学习Java的泛型
    文章图片


    总结 以上就是关于泛型和通配符的介绍,泛型可以在类、接口、方法中使用,分别简称之泛型类、泛型接口、泛型方法,并且通过泛型实现数据类型的任意化,即灵活、又有安全性,易于维护。在编译过中,对于正确检验泛型结果后,会将泛型的相关信息擦出,也就是说,成功编译过后的class文件中是不包含任何泛型信息的。泛型信息不会进入到运行时阶段。
    总而言之,泛型是在编译的时候检查类型安全,所有的强制转换都是自动和隐式的,提高代码的重用率。并且消除强制类型转换,在传入参数的时候就已经限制的类型。
    本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!

      推荐阅读