Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Java中泛型的介绍和使用

Java中泛型的介绍和使用

作者头像
框架师
发布于 2019-12-02 13:08:31
发布于 2019-12-02 13:08:31
1K00
代码可运行
举报
文章被收录于专栏:墨白的Java基地墨白的Java基地
运行总次数:0
代码可运行

!今天主要给大家说下Java中泛型的使用以及泛型是什么?下面请看正文

  • 泛型概述

我们都知道Java里的集合(list,map....)是可以存放任意对象的,只要把对象存储集合后,那么这时他们都会被提升成Object类型。当我们在取出每一个对象,并且进行相应的操作,这时必须采用类型转换。看下面这段示例代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package com.mobaijun;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;

public class GeneriDemo1{
    public static void main(String[] args) {
        Collection coll = new ArrayList();
        coll.add("mobai");
        coll.add("墨白");
        coll.add(5);//由于集合没有做任何限定,任何类型都可以给其中存放
        Iterator it = coll.iterator();
        while (it.hasNext()) {
            //需要打印每个字符串的长度,就要把迭代出来的对象转成String类型
            String str = (String) it.next();
            System.out.println(str.length());
        }
    }
}

我们可以看到控制台打印结果是一个异常,异信息如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String

这个Demo在运行时发生了java.lang.ClassCastException,找不到主类或无法加载类。为什么会发生类型转换异常呢?我们来分析下:由于集合中什么类型的元素都可以存储。导致取出时强转引发运行时 ClassCastException。怎么来解决这个问题呢?

Collection虽然可以存储各种对象,但实际上通常Collection只存储同一类型对象。例如都是存储字符串对象。因此 在JDK5之后,新增了泛型(Generic)语法,让你在设计API时可以指定类或方法支持泛型,这样的话我们使用API的时候 也变得更为简洁,并得到了编译时期的语法检查。

一句话概述就是:泛型可以在类或方法中预支地使用未知的类型

  • 那么使用泛型有什么好处呢?

看下面代码体验一下泛型带给我们的好处:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package com.mobaijun;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;

public class GeneriDemo2 {
    public static void main(String[] args) {
        Collection<String> list = new ArrayList<String>();
        list.add("mobai");
        list.add("墨白");
        // list.add(5);
        // 当集合明确类型后,存放类型不一致就会编译报错        
        // 集合已经明确具体存放的元素类型,那么在使用迭代器的时候,迭代器也同样会知道具体遍历元素类型
        Iterator<String> it = list.iterator();
        while (it.hasNext()) {
            String str = it.next();
            //当使用Iterator<String>控制元素类型后,就不需要强转了。获取到的元素直接就是String类 型            
            System.out.println(str.length());
        }
    }
}
  • 泛型的定义与使用

泛型,用来灵活地将数据类型应用到不同的类、方法、接口当中。将数据类型作为参数进行传递。

泛型的定义格式:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
修饰符 class 类名<代表泛型的变量> {  }

示例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class ArrayList<E> {
    public boolean add(E e) {
    }
    public E get(int index) {
    }
}

泛型在定义的时候不具体,使用的时候才变得具体。在使用的时候确定泛型的具体数据类型。

使用泛型:即什么时候确定泛型。

在创建对象的时候确定泛型

例如, ArrayList<String> list = new ArrayList<String>();

此时,变量E的值就是String类型,那么我们的类型就可以理解为

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class ArrayList<String> {
    public boolean add(String e) {
    }
    public String get(int index) {
    }
}

再例如, ArrayList<Integer> list = new ArrayList<Integer>();

此时,变量E的值就是Integer类型,那么我们的类型就可以理解为:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class ArrayList<Integer> {
    public boolean add(Integer e) {
    }
    public Integer get(int index) {
    }
}
  • 泛型方法

示例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class MyGenericMethod {
    public <MVP> void show(MVP mvp) {
        System.out.println(mvp.getClass());
    }

    public <MVP> MVP show2(MVP mvp) {
        return mvp;
    }
}

在方法调用的时候,明确泛型的类型:代码如下

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package com.mobaijun;

public class GeneriDemo3 {
    public static void main(String[] args) {
        MyGenericMethod my = new MyGenericMethod();
        my.show("墨白");
        my.show(18);
        my.show(178.0);
    }
}
  • 泛型的接口

示例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package com.mobaijun;

public interface MyGenericInterface<E> {
    public abstract void add(E e);
    public abstract E gitE();
}

下面我们来看下如果要定义一个类该如何使用泛型:

示例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package com.mobaijun;

public class MyImpl implements MyGenericInterface<String>{
    @Override
    public void add(String s) {
        // 省略
    }

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

在类开始的时候我们以及定义了泛型的类型,那么这个时候泛型e的值就是String;

  • 如果始终不确定泛型的类型,可以直到创建对象时,确定泛型的类型
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package com.mobaijun;

/**
 * 不确定泛型的类型
 * @param <E>
 */
public class MyImpl2<E> implements MyGenericInterface<E> {

    @Override
    public void add(E e) {

    }

    @Override
    public E gitE() {
        return null;
    }
}
  • 在创建对象的时候确定泛型的类型
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package com.mobaijun;

/**
 * 创建对象的时候确定泛型的类型
 */
public class GenericInterface  {
    public static void main(String[] args) {
        MyImpl2<String> my = new MyImpl2<String>();
        my.add("墨白");
    }
}
  • 泛型还有最后一个知识点,那就是泛型的通配符

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

通配符的基本使用 有如下几点

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

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

示例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package com.mobaijun;

import java.util.ArrayList;
import java.util.Collection;

public class MyDemo {
    public static void main(String[] args) {
        Collection<Integer> c = new ArrayList<Integer>();
       getElement(c);

       Collection<String> c2 = new ArrayList<String>();
       getElement(c2);
    }

    /**
     * 统配符中的?代表着任意类型数据
     * 需要注意的是泛型是不存在继承关系的,
     *      Collection<Object> c2 = new ArrayList<String>();错误示例
     * @param collection
     */
    public static void getElement(Collection<?> collection){

    }
}
  • 通配符高级使用

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

泛型的上限:

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

泛型的下限:

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

举个例子,比如:现已知Object类,String 类,Number类,Integer类,其中Number是Integer的父类

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package com.mobaijun;

import java.util.ArrayList;
import java.util.Collection;

public class MyDemo4 {
    public static void main(String[] args) {
        Collection<Integer> c1 = new ArrayList<Integer>();
        Collection<String> c2 = new ArrayList<String>();
        Collection<Number> c3 = new ArrayList<Number>();
        Collection<Object> c4 = new ArrayList<Object>();

        getElementOne(c1);
        getElementOne(c2);// 编译报错
        getElementOne(c3);
        getElementOne(c4);// 编译报错

        getElementTow(c1);// 编译报错
        getElementTow(c2);// 编译报错
        getElementTow(c3);
        getElementTow(c4);
    }
    // 泛型的上限:此时的泛型?,必须是Number类型或者Number类型的子类
    public static void getElementOne(Collection<? extends Number> collection){}
    // 泛型的下限:此时的泛型?,必须是Number类型或者Number类型的父类
    public static void getElementTow(Collection<? super Number> collection){}
}

以上就是泛型的全部内容了,如果有帮助到你,请在文末点个在看,祝你有个美好的一天,加油!!!

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-11-30,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 框架师 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
JDK1.9-泛型
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
cwl_java
2019/12/03
3980
Java中的泛型讲解【最详细】【精美版】
在前面学习集合时,我们都知道集合中是可以存放任意对象的,只要把对象存储集合后,那么这时他们都会被提升成Object类型。当我们在取出每一个对象,并且进行相应的操作,这时必须采用类型转换。
编程张无忌
2022/10/04
3510
Java中的泛型讲解【最详细】【精美版】
【Java_12】Collection、泛型
一、Collection 1. 集合框架 Collection - List - ArrayList ··· - Set - TreeSet ··· 2. 常用方法 ① add(E e) 将数据添加到集合。 ② remove(E e) 将数据从集合中移除 ③ contains(E e) 判断集合中是否包含该数据 ④ size() 获取集合的长度 ⑤ clear() 清空集合 ⑥ toAr
用户8250147
2021/02/04
2710
java_Collection、Map、泛型的使用
集合按照其存储结构可以分为两大类,分别是 单列集合 java.util.Collection 双列集合 java.util.Map
咕咕星
2020/08/19
1.2K0
java_Collection、Map、泛型的使用
【Java】08 集合
   Java 泛型的参数只可以代表类,不能代表个别对象。由于 Java 泛型的类型参数的实际类型在编译时会被消除,所以无法在运行时得知其类型参数的类型。Java 编译器在编译泛型时会自动加入类型转换的编码,故运行速度不会因为使用泛型而加快。
Demo_Null
2020/09/28
4210
【Java】08 集合
Java学习笔记——新特性-泛型
泛型,JDK1.5新加入的,解决数据类型的安全性问题,其主要原理是在类声明时通过一个标识表示类中某个属性的类型或者是某个方法的返回值及参数类型。这样在类声明或实例化时只要指定好需要的具体的类型即可。
梦飞
2022/06/23
2690
java基础之泛型
泛型 术语 "?"通配符 通配符的扩展 自定义泛型方法 "擦除"实例 类型参数的类型推断 自定义泛型类 泛型方法和泛型类的比较 泛型和反射 通过反射获得泛型的实际类型参数 本文对泛型的基本
xiangzhihong
2018/02/01
1.1K0
java基础之泛型
第50节:Java当中的泛型
这就存在一个问题,如果集合存储元素时,而且存储对象有很多,而且对象类型不相同,就很容易导致隐患。
达达前端
2019/07/03
7170
第50节:Java当中的泛型
Java(集合②)
Iterator迭代器是一个接口,我们无法直接使用,需要使用Iterator接口接口的实现类对象,获取方式比较特殊;
全栈开发日记
2022/05/12
3900
Java学习笔记(三):集合类与常用数据结构的典型用法
foochane :https://foochane.cn/article/2019122801.html 1 Collection集合 1.1 集合概述 在前面基础班我们已经学习过并使用过集合A
foochane
2020/02/13
9570
JAVAEE工程师零基础学习教程之泛型类和File类
​ 在前面学习集合时,我们都知道集合中是可以存放任意对象的,只要把对象存储集合后,那么这时他们都会被提升成Object类型。当我们在取出每一个对象,并且进行相应的操作,这时必须采用类型转换。
张哥编程
2024/12/13
1570
16(02)总结泛型
2:泛型(掌握) (1)泛型概述 是一种把明确类型的工作推迟到创建对象或者调用方法的时候才去明确的特殊的类型。 (2)格式: <数据类型> 注意:该数据类型只能是引用类型。(类,接口,数组) (3)好处: A:把运行时期的问题提前到了编译期间 B:避免了强制类型转换 C:优化了程序设计,解决了黄色警告线问题,让程序更安全 package cn.itcast_01; import java.util.ArrayList; import java.util.Iterator; /* * Ar
Java帮帮
2018/03/15
6490
Java基础(二十):泛型
Java微观世界
2025/01/21
2140
Java基础(二十):泛型
【Java】Collection集合&泛型
👀专栏介绍 【Java】 目前主要更新Java,一起学习一起进步。 👀本期介绍 本期主要介绍Collection集合&泛型 文章目录 第一章 Collection集合 1.1 集合概述 1.2 集合框架 1.3 Collection 常用功能 第二章 Iterator迭代器 2.1 Iterator接口 2.2 迭代器的实现原理 2.3 增强for 第三章 泛型 3.1 泛型概述 3.2 使用泛型的好处 3.3 泛型的定义与使用 定义和使用含有泛型的类 含有泛型的方法 含有泛型的接口 3.4 泛型通配符
陶然同学
2023/02/27
5680
【Java】Collection集合&泛型
聊聊JDK泛型那些事儿
在Java SE 1.5之前,没有泛型的情况的下,通过对类型Object的引用来实现参数的“任意化”,“任意化”带来的缺点是要做显式的强制类型转换,而这种转换是要求开发者对实际参数类型可以预知的情况下进行的。对于强制类型转换错误的情况,编译器可能不提示错误,在运行的时候才出现异常,这是一个安全隐患。 
孟君
2019/08/28
4040
聊聊JDK泛型那些事儿
JAVA入门学习六
描述: 集合的由来数组长度是固定,当添加的元素超过了数组的长度时需要对数组重新定义太麻烦,java内部给我们提供了集合类能存储任意对象,长度是可以改变的,随着元素的增加而增加,随着元素的减少而减少;
全栈工程师修炼指南
2020/10/23
6380
JAVA入门学习六
Java-Java5.0泛型解读
Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。
小小工匠
2021/08/17
5830
Collection集合
  集合框架集大致分为两大系列:一个是Collection系列,另一个是Map系列。
别团等shy哥发育
2023/02/25
1.1K0
Collection集合
Java泛型基础(二)泛型接口泛型方法# 总结
泛型可以应用于同一个类,该类可以针对多种类型使用,例如构建一个RedisTemplateService组件,用于处理当前应用中所有对象的缓存操作。这篇文章主要介绍泛型应用于接口、方法和匿名内部类的一些知识点和使用案例,也包括《Java编程思想》中对应的练习题的解读。
阿杜
2018/08/06
3.6K0
一文打通java泛型
集合容器类在设计阶段/声明阶段不能确定这个容器到底实际存的是什么类型的对象,所以在JDK1.5之前只能把元素类型设计为Object,JDK1.5之后使用泛型来解决。因为这个时候除了元素的类型不确定,其他的部分是确定的,例如关于这个元素如何保存,如何管理等是确定的,因此此时把元素的类型设计成一个参数,这个类型参数叫做泛型。Collection,List,ArrayList 这个就是类型参数,即泛型
一个风轻云淡
2023/10/15
2390
一文打通java泛型
相关推荐
JDK1.9-泛型
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验