详细全面解析Java泛型

1.概述

作为一个面向对象的编程语言,Java可以通过实现一些类,作为我们各种需求的一个模板,方便我们的使用。但有时候,这个类的范围可能比我们想要的范围要大,我们只想限定于满足类的某些对象,那这样的情况下,泛型的概念就被提出来了(非官方解释,方便理解)。

举个例子:比如我们我们生活中的车,它可以作为一个类,但是车其实又有很多种,包括货车,轿车,大巴车等等,而其中的轿车外观差不多,但是又属于不同的品牌,这些品牌有很多不一样的地方,这里我们可以把轿车的品牌看作是泛型(类似于标签)

通过上面的解释,泛型的概念就比较清晰了,就是一种“类型参数”,所谓类型参数可以理解为将类型由原来的具体的类型进行参数化,类似于方法中的变量参数,此时类型也定义成参数形式(可以称之为类型形参),然后在使用/调用时传入具体的类型(类型实参)。

泛型的优点,不仅仅是上面提到的,其还有下面的优点::

类型安全: 提高Java 程序的类型安全(泛型的主要目标)。
通过知道使用泛型定义的变量的类型限制,编译器可以验证类型假设。
消除强制类型转换:消除源代码中的许多强制类型转换。
这使得代码的可读性更高了,并且还减少了错误
上面说到了泛型在类中的使用,其实泛型的使用远不止于此,其还可以在在接口、方法中使用。下面就对这些分别进行介绍

2.泛型类

所谓泛型类就是把当我们在声明类时,类中的有些成员的类型并不是确定,然后我们可以把泛型定义在类上,当使用该类的时候,再把不确定成员的类型明确下来。

语法格式:

【修饰符】 class 类名<类型变量列表>{
    //类体
}

注: <类型变量列表>:可以是一个或多个类型变量,一般都是使用单个的大写字母表示。例如:、<K,V>等。

<类型变量列表>中的类型变量不能用于静态成员上。

泛型类的使用:

使用这种类似于参数化类型的类时,在创建类的对象时候,我们需要注意:

  • 指定类型变量对应的实际类型参数
  • 实际类型参数必须是引用数据类型,不能是基本数据类型

注:指定泛型实参时,必须左右两边一致,不存在多态现象(右边的可以省略不写)

代码示例:

泛型类的声明与使用:

public class Demo1 {
    public static void main(String[] args) {

        //泛型类的使用(<T>里面只能是引用类型)
        Student<Double> student1 = new Student<>("学生1",99.5);
        Student<String> student2 = new Student<>("学生2","优秀");
        Student<Character> student3 = new Student<>("学生3",'A');

        //输出结果
        System.out.println(student1);
        System.out.println(student2);
        System.out.println(student3);
    }
}

//泛型类的声明
class Student<T> { //<T>这个就是泛型类的类型参数
    private String name;
    private T score; //使用泛型,定义分数(分数可能有double类型(99.5)、字符串类型(优秀)、字符类型(‘A')等)

    //构造方法
    public Student() {
    }

    public Student(String name, T score) {
        this.name = name;
        this.score = score;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", score=" + score +
                '}';
    }

}

2.1泛型接口

泛型接口和泛型类关系,就像接口和类的关系一样。 这里不多说。

语法格式:

【修饰符】 interface 接口名<类型变量列表>{
     
}

注: <类型变量列表>:可以是一个或多个类型变量,一般都是使用单个的大写字母表示。例如:、<K,V>等。

<类型变量列表>中的类型变量不能用于静态成员上。

2.2泛型接口的使用

使用这种类似于参数化类型的接口时,我们需要注意:

指定类型变量对应的实际类型参数
实际类型参数必须是引用数据类型,不能是基本数据类型
代码示例

 泛型接口的声明与使用:

public class Demo1 {
    public static void main(String[] args) {

        //泛型类的使用(<T>里面只能是引用类型)
        Student<Double> student1 = new Student<>("学生1",99.5);

        //使用泛型接口
        student1.print("学生1",99.5);
    }
}

//泛型类的声明
class Student<T> implements Print<String,T>{ //<T>这个就是泛型类的,后面<String,T>是接口,多个类型变量
    private String name;
    private T score; //使用泛型
    //构造方法
    public Student() {
    }

    public Student(String name, T score) {
        this.name = name;
        this.score = score;
    }

    //重写接口的方法
    @Override
    public void print(String s, T t) {
        System.out.println("学生姓名:"+ this.name);
        System.out.println("学生成绩:"+ this.score);
    }
}

//泛型接口的声明
interface Print <T,V>{
    //定义一个打印函数,可以打印学生姓名和成绩
    public void print(T t, V v);

}

3.类型变量的上限和下限

前面说到,我们可以使用泛型类型参数,这样等我们进行实际使用的时候,我们可以任意使用类型,但如果想只使用某一系列的类型,泛型也是可以实现的。这就是我们说的类型变量的上限和类型变量的下限。下面进行分别介绍。

3.1类型变量的上限

如果泛型类定义了类型变量的上限,那么该泛型类实际的类型只能是该上限类型或者其子类类型。

语法格式:

泛型类和泛型方法的用法是一样的,后面都不再做区分。

<类型变量  extends 上限1 & 上限2> //上限可以有多个

注:如果多个上限中有类有接口,那么只能有一个类,而且必须写在最左边。接口的话,可以多个。
如果在声明<类型变量>时没有指定上限,默认上限是java.lang.Object。

代码示例:

类型变量的上限:

public class Demo2 {

    public static void main(String[] args) {
        Test<Double> test1 = new Test<>(77.5); //double类
//        Test<String> test2 = new Test<String>(); 不是数字类的子类
        Test<Integer> test3 = new Test<>(18);

        test1.print(77.5);
        test3.print(18);
    }
}

class Test<T extends Number >{ //数字类上限,只能使用数字类及其子类
    private T num;

    public Test() {
    }

    public Test(T num) {
        this.num = num;
    }

    public void print(T num){ //测试方法
        System.out.println(num);
    }
}

3.2类型变量的下限

如果泛型类定义了类型变量的下限,那么该泛型类实际的类型只能是该下限类型或者其父类类型。

语法格式:

<?  super E > // ? 代表接收E类型或者E的父类型的元素

? 是泛型类中的通配符(下面会讲到,可以先看下面的再回来看这个)

代码示例:

/*
    <? super 下限>
 */

public class Demo5 {
    public static void main(String[] args){
        C<String> c=new C<>();
        c.setT("<? super 下限>");
        fun1(c);

    }

       //测试函数,泛型类使用了下限
    public static void fun1(C<? super String> c){      //接受的数据类型只能为String、Object
   
        System.out.println(c.getT()); //输入测试
    }
}

class C<T>{

    private T t;

    public T getT() {
        return t;
    }
    public void setT(T t) {
        this.t = t;
    }
}

4.泛型方法

鉴于某个方法定义时,想要自己定义类型变量或者在某个静态方法中定义类型变量的需求,JDK还提供了泛型方法的支持。即可以在某个方法定义时,自定以<类型变量>

注:前面说到类和接口上的类型形参是不能用于静态方法

语法格式:

【修饰符】 <类型变量列表> 返回值类型 方法名(【形参列表】)【throws 异常列表】{
    //方法体
}

注:- <类型变量列表>:可以是一个或多个类型变量,一般都是使用单个的大写字母表示。例如: < T >、<K,V>等。
<类型变量>同样也可以指定上限

代码示例:

/*
    泛型方法
 */

public class Demo3 {
    public static void main(String[] args) {
        Test1 test = new Test1(); //创建测试对象

        test.print(12); //测试
        test.print(12.5); //测试

    }
}

class Test1{

    public <T extends Number> void print(T t){ //泛型方法,可以设置上限
        System.out.println("这是一个泛型方法,测试类型:" + t);
    }
}

5.泛型擦除

泛型擦除只是在编译阶段才会有的,在实际运行阶段类型已经确定了,这个时候就没有泛型的概念了(JVM并不知道泛型的存在)。这个从有泛型信息到没有泛型信息的过程称之为“泛型擦除”。

其擦除规则如下:

  • 若泛型类型没有指定具体类型,用Object作为原始类型;
  • 若有限定类型< T exnteds XClass >,使用XClass作为原始类型;
  • 若有多个限定< T exnteds XClass1 & XClass2 >,使用第一个边界类型XClass1作为原始类型;

6.类型通配符

通配符的意思是可以指代很多类型。这个主要使用在当我们在声明方法时,不确定该泛型实际类型的情况。

类型通配符有三种:

  • <?> 任意类型
  • <? extends 上限>
  • <? super E>

下面对这三种通配符分别进行介绍:

<?> 任意类型

当泛型使用这种 类型通配符的时候,表示可以使用任意类型

代码示例:

/*
    类型通配符
 */

public class Demo4 {

    public static void main(String[] args) {
        // 语文老师使用时:
        StudentInfo<String> stu1 = new StudentInfo<String>("张三", "良好");

        // 数学老师使用时:
        StudentInfo<Double> stu2 = new StudentInfo<Double>("张三", 90.5);

        // 英语老师使用时:
        StudentInfo<Character> stu3 = new StudentInfo<Character>("张三", 'C');

        StudentInfo<?>[] arr = new StudentInfo[3]; //使用通配符
        arr[0] = stu1;
        arr[1] = stu2;
        arr[2] = stu3;

        StudentInfoPrint.print(arr); //打印输出结果
    }

}

//学生类是一个参数化的泛型类
class StudentInfo<T>{
    private String name;
    private T score;

    public StudentInfo() {
        super();
    }
    public StudentInfo(String name, T score) {
        super();
        this.name = name;
        this.score = score;
    }
    @Override
    public String toString() {
        return "姓名:" + name + ", 成绩:" + score;
    }
}

//学生信息打印类
class StudentInfoPrint {
    //泛型方法,使用通配符
    public static void print(StudentInfo<?>[] arr) {
        for (int i = 0; i < arr.length; i++) {
            System.out.println(arr[i]);
        }
    }
}

<? extends 上限>

? 代表接收E类型或者E的子类型的元素

代码示例

可参考上面的类型变量的上限代码

<? super E>

? 代表接收E类型或者E的父类型的元素

代码示例

可参考上面的类型变量的下限代码

到此这篇关于详细全面解析Java泛型的文章就介绍到这了,更多相关Java泛型内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 关于Java反射给泛型集合赋值问题

    泛型 Java泛型简单描述下: 比如创建一个List集合,我想在里边只放Student对象信息,就需要写成 List<Student> studentList = new ArrayList(); 这个时候List里面就只能放入Student类型的值,如果强行放入其他类型(比如说Integer)的就会提示错误信息: java.lang.ClassCastException: java.lang.Integer cannot be cast to com.entity.Student 按照这个结

  • 这个Java泛型不太正经

    目录 一.前言 二.泛型 三.泛型定义的格式: 什么是引用类型? 泛型的好处是: 四.泛型类 五.泛型方法 六.泛型接口 七.类型通配符 总结 一.前言 泛型在java中有很重要的地位,在实际开发中用处也很大. 二.泛型 泛型:是jdk5中引入的特性,他提供编译是类型是类型的安全检测机制,这个机制允许在编译时检测到非法的类型,它的本质是参数化类型,也就是所操作的数据类型不变指定为一个参数 将原来具体的类型参数化,然后再使用/调用的时候传入具体的参数 泛型的类型: ①:泛型类 ②:泛型方法 ③:泛

  • Java 泛型详解与范例

    目录 一.泛型的使用 二.泛型类的定义-类型边界 三.类型擦除 四.泛型类的使用-通配符 五.泛型方法 六.泛型的限制 一.泛型的使用 前面我们学集合的时候,简单的说过泛型的使用.如下: ArrayList<Integer> list = new ArrayList<>(); Queue<Integer> queue = new LinkedList<>(); 那么使用是这样的简单,该注意什么? 尖括号里的类型,只能写引用类型 基础数据类型的话,就需要写相应

  • Java中关于泛型、包装类及ArrayList的详细教程

    目录 一.泛型 1.1 泛型类的定义 1.2 泛型类的使用 1.3 泛型总结 二.包装类 2.1基本数据类型和包装类直接的对应关系 2.2 包装类的使用,装箱(boxing)和拆箱(unboxing) 2.3 自动装箱(autoboxing)和自动拆箱(autounboxing) 三.List 的使用 3.1 ArrayList简介 3.3 ArrayList的遍历 3.4 ArrayList的常见操作 3.4.1 删除 index 位置元素(remove) 3.4.1尾插(add) 3.4.2

  • Java 中泛型 T 和 ? 的区别详解

    目录 泛型中 T 类型变量 和 ? 通配符 区别 Generic Types 类型变量 用法 2.声明通用的方法 – 泛型方法: 有界类型参数 Wildcards 通配符 1.上界通配符:? extend 上界类型 2.无界通配符:? 3.下界通配符:? super 子类 类型擦除 泛型中 T 类型变量 和 ? 通配符 区别 定义不同 :T 是类型变量,? 是通配符 使用范围不同: ? 通配符用作 参数类型.字段类型.局部变量类型,有时作为返回类型(但请避免这样做) T 用作 声明类的类型参数.

  • Java泛型之类型擦除实例详解

    目录 前言 泛型是什么? 泛型的定义和使用 泛型类 泛型方法 泛型类与泛型方法的共存现象 泛型接口 通配符 ? 无限定通配符 <?> <? extends T> 类型擦除 类型擦除带来的局限性 泛型中值得注意的地方 Java 不能创建具体类型的泛型数组 泛型,并不神奇 总结 前言 泛型,一个孤独的守门者. 大家可能会有疑问,我为什么叫做泛型是一个守门者.这其实是我个人的看法而已,我的意思是说泛型没有其看起来那么深不可测,它并不神秘与神奇.泛型是 Java 中一个很小巧的概念,但同时

  • 详细全面解析Java泛型

    1.概述 作为一个面向对象的编程语言,Java可以通过实现一些类,作为我们各种需求的一个模板,方便我们的使用.但有时候,这个类的范围可能比我们想要的范围要大,我们只想限定于满足类的某些对象,那这样的情况下,泛型的概念就被提出来了(非官方解释,方便理解). 举个例子:比如我们我们生活中的车,它可以作为一个类,但是车其实又有很多种,包括货车,轿车,大巴车等等,而其中的轿车外观差不多,但是又属于不同的品牌,这些品牌有很多不一样的地方,这里我们可以把轿车的品牌看作是泛型(类似于标签) 通过上面的解释,泛

  • 解析Java 泛型什么情况下不能使用

    一.前言 Java泛型来保证类型安全,防止在运行时发生类型转换异常,让类型参数化,提高了代码的可读性和重用率.但是有些情况下泛型也是不允许使用的,以下是不能使用泛型的一些场景. 二. 什么情况下不能使用Java泛型 1 不能使用泛型的形参创建对象. T o=new T(); // 不允许 2 在泛型类中,不能给静态成员变量定义泛型 Java 中的静态类型随着类加载而实例化,此时泛型的具体类型并没有声明.同时因为静态变量作为所有对象的共享变量,只有类实例化或者方法调用时才能确定其类型.如果是泛型类

  • Java 泛型考古 泛型擦除 包装类详细解析

    目录 一. 什么是泛型 二. 为什么要有泛型 ? 示例 三.泛型考古 四.泛型擦除 五.包装类 六.装箱拆箱 一. 什么是泛型 泛型(generic type)其本质是将类型参数化,也就是说所操作的数据类型被指定为一个参数这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型接口.泛型方法. 二. 为什么要有泛型 ? 之前写过MyArrayList顺序表,这个类当时自己在实现的时候只能用一种类型来表示,也就是用的时候自己实现的MyArrayList只能应用于一种类型,要想应用于其他类型

  • JAVA泛型的继承和实现、擦除原理解析

    这篇文章主要介绍了JAVA泛型的继承和实现.擦除原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 很多的基础类设计会采用泛型模式,有些应用在使用的时候处于隔离考虑,会进行继承,此时子类如何继承泛型类就很讲究了,有些情况下需要类型擦除,有些情况下不需要类型擦除,但是大多数情况下,我们需要的是保留父类的泛型特性.因为类型被擦除后,得到的对象会是Object,此时会导致编码或反序列化失败. 如下所示: 如果子类也继承父类的泛型化,则类型不会被擦

  • Java泛型的用法及T.class的获取过程解析

    这篇文章主要介绍了Java泛型的用法及T.class的获取过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 胡乱总结泛型的四点作用: 第一是泛化,可以拿个T代表任意类型. 但GP是被C++严苛的静态性逼出来的,落到Java.C#这样的花语平原里----所有对象除几个原始类型外都派生于Object,再加上Java的反射功能,Java的Collection库没有范型一样过得好好的. 第二是泛型 + 反射,原本因为Java的泛型拿不到T.cla

  • java泛型常用通配符实例解析

    这篇文章主要介绍了java泛型常用通配符实例解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 今天在看ArrayList原码是看到这样的一个符号,好奇怪. ?表示通配符,表示的意思是匹配E或E的子类,具体类型未知. 1.限定通配符 编写一个类似于ArrayList的动态数据 public class Gys<T> { private final static int default_capacity =10; private int end

  • Java 泛型全解析

    泛型简介 什么是泛型? 参化类型,数是JDK1.5的新特性.(定义泛型时使用参数可以简单理解为形参),例如List<E>,Map<K,V> 编译时的一种类型,此类型仅仅在编译阶段有效,运行时无效.例如List<String>在运行时String会被擦除,最终系统会认为都是Object. 为什么要使用泛型? 泛型是进行类型设计或方法定义时的一种约束规范,基于此规范可以: 提高编程时灵活性(有点抽象,后续结合实例理解). 提高程序运行时的性能.(在编译阶段解决一些运行时需要

  • 详细分析Java 泛型的使用

    一.泛型的简介 1.为什么要使用泛型? 一般使用在集合上,比如现在把一个字符串类型的值放入到集合里面,这个时候,这个值放到集合之后,失去本身的类型,只能是object类型.这时,如果想要对这个值进行类型转换,很容易出现类型转换错误,怎么解决这个问题,可以使用泛型来解决. 2.在泛型里面写是一个对象,String 不能写基本的数据类型 比如int,要写基本的数据类型对应的包装类 基本数据类型 对应包装类 基本数据类型 对应包装类 byte Byte short Short int Integer

  • Java 泛型超详细入门讲解

    目录 1.什么是泛型? 2.泛型是怎么编译的 泛型的编译机制:擦除机制 1.什么是泛型? 泛型其实就是将类型作为参数传递,泛型允许程序员在编写代码时使用一些以后才指定的类型 ,在实例化该类时将想要的类型作为参数传递,来指明这些类型. 为什么要引入泛型? 例如:自己实现一个顺序表 public class MyArrayList { public int[] elem; public int usedSize; public MyArrayList() { this.elem = new int[

  • Java 泛型超详细入门讲解

    目录 1.什么是泛型? 2.泛型是怎么编译的 泛型的编译机制:擦除机制 1.什么是泛型? 泛型其实就是将类型作为参数传递,泛型允许程序员在编写代码时使用一些以后才指定的类型 ,在实例化该类时将想要的类型作为参数传递,来指明这些类型. 为什么要引入泛型? 例如:自己实现一个顺序表 public class MyArrayList { public int[] elem; public int usedSize; public MyArrayList() { this.elem = new int[

随机推荐