Java 泛型详解(超详细的java泛型方法解析)


Posted in Java/Android onJuly 02, 2021
目录
  • 2. 什么是泛型
  • 3. 使用泛型的好处
  • 4. 泛型的使用
    • 4.1 泛型类
    • 4.2 泛型方法
    • 4.3 泛型接口
  • 5. 泛型通配符
    • 5.1 通配符基本使用
    • 5.2 通配符高级使用
  • 6. 总结

1. 为什么使用泛型

早期的Object类型可以接收任意的对象类型,但是在实际的使用中,会有类型转换的问题。也就存在这隐患,所以Java提供了泛型来解决这个安全问题。

  • 来看一个经典案例:
public static void main(String[] args) {
        //测试一下泛型的经典案例
        ArrayList arrayList = new ArrayList();
        arrayList.add("helloWorld");
        arrayList.add("taiziyenezha");
        arrayList.add(88);//由于集合没有做任何限定,任何类型都可以给其中存放
        for (int i = 0; i < arrayList.size(); i++) {
            //需求:打印每个字符串的长度,就要把对象转成String类型
            String str = (String) arrayList.get(i);
            System.out.println(str.length());
        }
    }

运行这段代码,程序在运行时发生了异常:

Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String

发生了数据类型转换异常,这是为什么?

由于ArrayList可以存放任意类型的元素。例子中添加了一个String类型,添加了一个Integer类型,再使用时都以String的方式使用,导致取出时强制转换为String类型后,引发了ClassCastException,因此程序崩溃了。

这显然不是我们所期望的,如果程序有潜在的错误,我们更期望在编译时被告知错误,而不是在运行时报异常。而为了解决类似这样的问题(在编译阶段就可以解决),在jdk1.5后,泛型应运而生。让你在设计API时可以指定类或方法支持泛型,这样我们使用API的时候也变得更为简洁,并得到了编译时期的语法检查。

我们将第一行声明初始化ArrayList的代码更改一下,编译器就会在编译阶段就能够帮我们发现类似这样的问题。现在再看看效果。

ArrayList<String> arrayList = new ArrayList<>();
arrayList.add("helloWorld");
arrayList.add("taiziyenezha");
arrayList.add(88);// 在编译阶段,编译器就会报错

这样可以避免了我们类型强转时出现异常。

 

2. 什么是泛型

泛型:是一种把明确类型的工作推迟到创建对象或者调用方法的时候才去明确的特殊的类型。也就是说在泛型使用过程中,操作的数据类型被指定为一个参数,而这种参数类型可以用在类、方法和接口中,分别被称为泛型类泛型方法泛型接口

注意:一般在创建对象时,将未知的类型确定具体的类型。当没有指定泛型时,默认类型为Object类型。

 

3. 使用泛型的好处

  • 避免了类型强转的麻烦。
  • 它提供了编译期的类型安全,确保在泛型类型(通常为泛型集合)上只能使用正确类型的对象,避免了在运行时出现ClassCastException。

 

4. 泛型的使用

泛型虽然通常会被大量的使用在集合当中,但是我们也可以完整的学习泛型只是。泛型有三种使用方式,分别为:泛型类、泛型方法、泛型接口。将数据类型作为参数进行传递。

 

4.1 泛型类

泛型类型用于类的定义中,被称为泛型类。通过泛型可以完成对一组类的操作对外开放相同的接口。最典型的就是各种集合框架容器类,如:List、Set、Map。

  • 泛型类的定义格式:

修饰符 class 类名<代表泛型的变量> { }

怕你不清楚怎么使用,这里我还是做了一个简单的泛型类:

/**
 * @param <T> 这里解释下<T>中的T:
 *           此处的T可以随便写为任意标识,常见的有T、E等形式的参数表示泛型
 *           泛型在定义的时候不具体,使用的时候才变得具体。
 *           在使用的时候确定泛型的具体数据类型。即在创建对象的时候确定泛型。
 */
public class GenericsClassDemo<T> {
    //t这个成员变量的类型为T,T的类型由外部指定
    private T t;
    //泛型构造方法形参t的类型也为T,T的类型由外部指定
    public GenericsClassDemo(T t) {
        this.t = t;
    }
    //泛型方法getT的返回值类型为T,T的类型由外部指定
    public T getT() {
        return t;
    }
}

泛型在定义的时候不具体,使用的时候才变得具体。在使用的时候确定泛型的具体数据类型。即:在创建对象的时候确定泛型。

例如:Generic<String> genericString = new Generic<String>("helloGenerics");

此时,泛型标识T的类型就是String类型,那我们之前写的类就可以这么认为:

public class GenericsClassDemo<String> {
    private String t;
    public GenericsClassDemo(String t) {
        this.t = t;
    }
    public String getT() {
        return t;
    }
}

当你的泛型类型想变为Integer类型时,也是很方便的。直接在创建时,T写为Integer类型即可:

Generic<Integer> genericInteger = new Generic<Integer>(666);

  • 注意: 定义的泛型类,就一定要传入泛型类型实参么?

并不是这样,在使用泛型的时候如果传入泛型实参,则会根据传入的泛型实参做相应的限制,此时泛型才会起到本应起到的限制作用。如果不传入泛型类型实参的话,在泛型类中使用泛型的方法或成员变量定义的类型可以为任何的类型。即跟之前的经典案例一样,没有写ArrayList的泛型类型,容易出现类型强转的问题。

 

4.2 泛型方法

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

  • 定义格式:

修饰符 <代表泛型的变量> 返回值类型 方法名(参数){ }

例如:

/**
     *
     * @param t 传入泛型的参数
     * @param <T> 泛型的类型
     * @return T 返回值为T类型
     * 说明:
     *   1)public 与 返回值中间<T>非常重要,可以理解为声明此方法为泛型方法。
     *   2)只有声明了<T>的方法才是泛型方法,泛型类中的使用了泛型的成员方法并不是泛型方法。
     *   3)<T>表明该方法将使用泛型类型T,此时才可以在方法中使用泛型类型T。
     *   4)与泛型类的定义一样,此处T可以随便写为任意标识,常见的如T、E等形式的参数常用于表示泛型。
     */
    public <T> T genercMethod(T t){
        System.out.println(t.getClass());
        System.out.println(t);
        return t;
    }

调用方法时,确定泛型的类型

public static void main(String[] args) {
    GenericsClassDemo<String> genericString  = new GenericsClassDemo("helloGeneric"); //这里的泛型跟下面调用的泛型方法可以不一样。
    String str = genericString.genercMethod("hello");//传入的是String类型,返回的也是String类型
    Integer i = genericString.genercMethod(123);//传入的是Integer类型,返回的也是Integer类型
}

这里我们可以看下结果:

class java.lang.String

hello

class java.lang.Integer
123

这里可以看出,泛型方法随着我们的传入参数类型不同,他得到的类型也不同。泛型方法能使方法独立于类而产生变化。

 

4.3 泛型接口

泛型接口与泛型类的定义及使用基本相同。泛型接口常被用在各种类的生产器中。

  • 定义格式
修饰符 interface接口名<代表泛型的变量> {  }

 

看一下下面的例子,你就知道怎么定义一个泛型接口了:

/**
 * 定义一个泛型接口
 */
public interface GenericsInteface<T> {
    public abstract void add(T t); 
}

使用格式

  • 1、定义类时确定泛型的类型
public class GenericsImp implements GenericsInteface<String> {
    @Override
    public void add(String s) {
        System.out.println("设置了泛型为String类型");
    }
}
  • 2、始终不确定泛型的类型,直到创建对象时,确定泛型的类型
public class GenericsImp<T> implements GenericsInteface<T> {
    @Override
    public void add(T t) {
        System.out.println("没有设置类型");
    }
}

确定泛型:

public class GenericsTest {
    public static void main(String[] args) {
        GenericsImp<Integer> gi = new GenericsImp<>();
        gi.add(66);
    }
}

 

5. 泛型通配符

当使用泛型类或者接口时,传递的数据中,泛型类型不确定,可以通过通配符<?>表示。但是一旦使用泛型的通配符后,只能使用Object类中的共性方法,集合中元素自身方法无法使用。

 

5.1 通配符基本使用

泛型的通配符:不知道使用什么类型来接收的时候,此时可以使用?,?表示未知通配符。

此时只能接受数据,不能往该集合中存储数据。

举个例子大家理解使用即可:

// ?代表可以接收任意类型
// 泛型不存在继承、多态关系,泛型左右两边要一样
//ArrayList<Object> list = new ArrayList<String>();这种是错误的
//泛型通配符?:左边写<?> 右边的泛型可以是任意类型
ArrayList<?> list1 = new ArrayList<Object>();
ArrayList<?> list2 = new ArrayList<String>();
ArrayList<?> list3 = new ArrayList<Integer>();

注意:泛型不存在继承、多态关系,泛型左右两边要一样,jdk1.7后右边的泛型可以省略

而泛型通配符?,右边的泛型可以是任意类型。

泛型通配符?主要应用在参数传递方面,让我们一起瞧瞧呗:

public static void main(String[] args) {
    ArrayList<Integer> list1 = new ArrayList<Integer>();
    test(list1);
    ArrayList<String> list2 = new ArrayList<String>();
    test(list2);
}
public static void test(ArrayList<?> coll){
}

嘿嘿,是不是见识到了通配符的厉害,可以传递不同类似进去方法中了!

 

5.2 通配符高级使用

之前设置泛型的时候,实际上是可以任意设置的,只要是类就可以设置。但是在JAVA的泛型中可以指定一个泛型的上限下限

泛型的上限:

  • 格式 类型名称 <? extends 类 > 对象名称
  • 意义只能接收该类型及其子类

泛型的下限:

  • 格式: 类型名称 <? super 类 > 对象名称
  • 意义: 只能接收该类型及其父类型

比如:现已知Object类,Animal类,Dog类,Cat类,其中Animal是Dog,Cat的父类

class Animal{}//父类
class Dog extends Animal{}//子类
class Cat extends Animal{}//子类

首先我们先看下,泛型的上限<? extends 类 >:

//        ArrayList<? extends Animal> list = new ArrayList<Object>();//报错
        ArrayList<? extends Animal> list2 = new ArrayList<Animal>();
        ArrayList<? extends Animal> list3 = new ArrayList<Dog>();
        ArrayList<? extends Animal> list4 = new ArrayList<Cat>();

可以看出,泛型的上限只能是该类型的类型及其子类。

Java 泛型详解(超详细的java泛型方法解析)

  • 我们再来看看泛型的下限<? super 类 >:
ArrayList<? super Animal> list5 = new ArrayList<Object>();
        ArrayList<? super Animal> list6 = new ArrayList<Animal>();
//        ArrayList<? super Animal> list7 = new ArrayList<Dog>();//报错
//        ArrayList<? super Animal> list8 = new ArrayList<Cat>();//报错

可以看出,泛型的下限只能是该类型的类型及其父类。

Java 泛型详解(超详细的java泛型方法解析)

  • 一般泛型的上限和下限也是用来参数的传递:

再比如:现已知Object类,String 类,Number类,Integer类,其中Number是Integer的父类

public static void main(String[] args) {
    Collection<Integer> list1 = new ArrayList<Integer>();
    Collection<String> list2 = new ArrayList<String>();
    Collection<Number> list3 = new ArrayList<Number>();
    Collection<Object> list4 = new ArrayList<Object>();
    getElement(list1);
    getElement(list2);//报错
    getElement(list3);
    getElement(list4);//报错
    getElement2(list1);//报错
    getElement2(list2);//报错
    getElement2(list3);
    getElement2(list4);
}
// 泛型的上限:此时的泛型?,必须是Number类型或者Number类型的子类
public static void getElement1(Collection<? extends Number> coll){}
// 泛型的下限:此时的泛型?,必须是Number类型或者Number类型的父类
public static void getElement2(Collection<? super Number> coll){}
118060630

学到这里,我们泛型也就学完了!

 

6. 总结

这篇文章就到这里了,如果这篇文章对你也有所帮助,希望您能多多关注三水点靠木的更多内容!

Java/Android 相关文章推荐
教你用Java在个人电脑上实现微信扫码支付
Jun 13 Java/Android
Spring Boot 整合 Apache Dubbo的示例代码
Jul 04 Java/Android
使用@Value值注入及配置文件组件扫描
Jul 09 Java/Android
Java生成读取条形码和二维码的简单示例
Jul 09 Java/Android
java设计模式--原型模式详解
Jul 21 Java/Android
Java 在生活中的 10 大应用
Nov 02 Java/Android
关于Spring配置文件加载方式变化引发的异常详解
Jan 18 Java/Android
关于EntityWrapper的in用法
Mar 22 Java/Android
SpringBoot2零基础到精通之数据库专项精讲
Mar 22 Java/Android
Android超详细讲解组件ScrollView的使用
Mar 31 Java/Android
Mybatis-plus配置分页插件返回统一结果集
Jun 21 Java/Android
Spring Cloud OpenFeign模版化客户端
Jun 25 Java/Android
SpringBoot集成Druid连接池连接MySQL8.0.11
Java使用httpRequest+Jsoup爬取红蓝球号码
使用Springboot实现健身房管理系统
Java Kafka 消费积压监控的示例代码
springboot临时文件存储目录配置方式
Jul 01 #Java/Android
Mybatis-plus在项目中的简单应用
Jul 01 #Java/Android
SpringBoot 拦截器妙用你真的了解吗
You might like
自制短波长线天线频率预选器 - 成功消除B2K之流的镜像
2021/03/02 无线电
PHP cron中的批处理
2008/09/16 PHP
PHP面向对象分析设计的经验原则
2008/09/20 PHP
国外PHP程序员的13个好习惯小结
2012/02/20 PHP
PHP弱类型语言中类型判断操作实例详解
2017/08/10 PHP
PHP实现笛卡尔积算法的实例讲解
2019/12/22 PHP
JS获取浏览器版本及名称实现函数
2013/04/02 Javascript
使用node.js半年来总结的 10 条经验
2014/08/18 Javascript
JavaScript设计模式初探
2016/01/07 Javascript
实例详解JSON数据格式及json格式数据域字符串相互转换
2016/01/07 Javascript
nodejs 中模拟实现 emmiter 自定义事件
2016/02/22 NodeJs
Vue.js绑定HTML class数组语法错误的原因分析
2016/10/19 Javascript
fckeditor部署到weblogic出现xml无法读取及样式不能显示问题的解决方法
2017/03/24 Javascript
微信小程序实战之自定义抽屉菜单(7)
2017/04/18 Javascript
SeaJS中use函数用法实例分析
2017/10/10 Javascript
vuex+axios+element-ui实现页面请求loading操作示例
2020/02/02 Javascript
node中短信api实现验证码登录的示例代码
2021/01/20 Javascript
[02:56]DOTA2亚洲邀请赛 VG出场战队巡礼
2015/02/07 DOTA
利用python3随机生成中文字符的实现方法
2017/11/24 Python
基于Django contrib Comments 评论模块(详解)
2017/12/08 Python
python 时间信息“2018-02-04 18:23:35“ 解析成字典形式的结果代码详解
2018/04/19 Python
Python反射和内置方法重写操作详解
2018/08/27 Python
解决pycharm每次新建项目都要重新安装一些第三方库的问题
2019/01/17 Python
Python企业编码生成系统之主程序模块设计详解
2019/07/26 Python
澳大利亚排名第一的在线酒类商店:MyBottleShop
2018/04/26 全球购物
Dillard’s百货官网:Dillards.com
2018/05/26 全球购物
团日活动策划书
2014/02/01 职场文书
校长寄语大全
2014/04/09 职场文书
幼儿园评语大全
2014/04/17 职场文书
简单的离婚协议书范本
2014/11/16 职场文书
岳麓书院导游词
2015/02/03 职场文书
大学生党员自我评价
2015/03/04 职场文书
2015年度信用社工作总结
2015/05/04 职场文书
追悼会答谢词范文
2015/09/29 职场文书
浅谈resultMap的用法及关联结果集映射
2021/06/30 Java/Android
详解Python中__new__方法的作用
2022/03/31 Python