泛型的通配符及擦除机制详解

 

目录

一、通配符解决什么问题

二、通配符上界

三、通配符下界  ​编辑

四、泛型类的擦除机制 


引言:

        在这篇文章中,我主要介绍前一篇泛型没介绍完整的泛型通配符和泛型的擦除机制Java中泛型的详细介绍

? 用于在泛型的使用,即为通配符

一、通配符解决什么问题

static class Message<T> {
        private T message ;
        public T getMessage() {
            return message;
        }
        public void setMessage(T message) {
            this.message = message;
        }
    }
    public class TestDemo {
        public static void main(String[] args) {
            Message<String> message = new Message<>() ;
            message.setMessage("比特就业课欢迎您");
            fun(message);
        }
        public static void fun(Message<String> temp){
            System.out.println(temp.getMessage());
        }
    }

以上程序会带来新的问题,如果现在泛型的类型设置的不是String,而是Integer.  

public class TestDemo {
        public static void main(String[] args) {
            Message<Integer> message = new Message() ;
            message.setMessage(99);
            fun(message); // 出现错误,只能接收String
        }
        public static void fun(Message<String> temp){
            System.out.println(temp.getMessage());
        }
    }

        我们需要的解决方案:可以接收所有的泛型类型,但是又不能够让用户随意修改。这种情况就需要使用通配符"?"来 处理

 示例:使用通配符

public class TestDemo {
        public static void main(String[] args) {
            Message<Integer> message = new Message() ;
            message.setMessage(55);
            fun(message);
        }
        // 此时使用通配符"?"描述的是它可以接收任意类型,但是由于不确定类型,所以无法修改
        public static void fun(Message<?> temp){
        //temp.setMessage(100); 无法修改!
            System.out.println(temp.getMessage());
        }
    }

二、通配符上界

<? extends 上界>
<? extends Number>//可以传入的实参类型是Number或者Number的子类

示例:

    class Food {
    }
    class Fruit extends Food {
    }
    class Apple extends Fruit {
    }
    class Banana extends Fruit {
    }
    class Message<T> { // 设置泛型
        private T message ;
        public T getMessage() {
            return message;
        }
        public void setMessage(T message) {
            this.message = message;
        }
    }
    public class TestDemo {
        public static void main(String[] args) {
            Message<Apple> message = new Message<>() ;
            message.setMessage(new Apple());
            fun(message);
            Message<Banana> message2 = new Message<>() ;
            message2.setMessage(new Banana());
            fun(message2);
        }
        // 此时使用通配符"?"描述的是它可以接收任意类型,但是由于不确定类型,所以无法修改
        public static void fun(Message<? extends Fruit> temp){
        //temp.setMessage(new Banana()); //仍然无法修改!
        //temp.setMessage(new Apple()); //仍然无法修改!
            System.out.println(temp.getMessage());
        }
    }
        此时无法在fun 函数中对 temp 进行添加元素,因为 temp 接收的是 Fruit 和他的子类,此时存储的元素应该是哪个子类无法确定。所以添加会报错!但是可以获取元素。

public static void fun(Message<? extends Fruit> temp){
        //temp.setMessage(new Banana()); //仍然无法修改!
        //temp.setMessage(new Apple()); //仍然无法修改!
        Fruit b = temp.getMessage();
        System.out.println(b);
    }

通配符的上界,不能进行写入数据,只能进行读取数据 

三、通配符下界  

<? super 下界>
<? super Integer>//代表 可以传入的实参的类型是Integer或者Integer的父类类型

示例:

class Food {
    }
    class Fruit extends Food {
    }
    class Apple extends Fruit {
    }
    class Plate<T> {
        private T plate ;
        public T getPlate() {
            return plate;
        }
        public void setPlate(T plate) {
            this.plate = plate;
        }
    }
    public class TestDemo {
        public static void main(String[] args) {
            Plate<Fruit> plate1 = new Plate<>();
            plate1.setPlate(new Fruit());
            fun(plate1);
            Plate<Food> plate2 = new Plate<>();
            plate2.setPlate(new Food());
            fun(plate2);
        }
        public static void fun(Plate<? super Fruit> temp){
            // 此时可以修改!!添加的是Fruit 或者Fruit的子类
            temp.setPlate(new Apple());//这个是Fruit的子类
            temp.setPlate(new Fruit());//这个是Fruit的本身
            //Fruit fruit = temp.getPlate(); 不能接收,这里无法确定是哪个父类
            System.out.println(temp.getPlate());//只能直接输出
        }
    }

 通配符的下界,不能进行读取数据,只能写入数据。

 

四、泛型类的擦除机制 

        Java 中的泛型是在编译时期进行类型检查的机制,但在编译后会被擦除。这意味着在运行时,Java 虚拟机不会知道泛型的具体类型,而是使用泛型类型的原始类型。这个过程被称为类型擦除。

        具体来说,编译器在编译泛型类或方法时,会将其中的类型参数用其上限(如果没有指定上限,则为 Object来替换,并插入必要的类型转换以确保类型安全。这样做是为了保持向后兼容性,因为泛型是在 Java 5 中引入的,而在此之前的版本中并没有泛型支持。

下面是一个简单的例子来说明泛型类的擦除机制:

import java.util.*;

public class GenericExample<T> {
    private T data;

    public GenericExample(T data) {
        this.data = data;
    }

    public T getData() {
        return data;
    }

    public static void main(String[] args) {
        GenericExample<String> stringExample = new GenericExample<>("Hello");
        GenericExample<Integer> integerExample = new GenericExample<>(123);

        // 编译后会擦除泛型类型
        System.out.println(stringExample.getClass()); // class GenericExample
        System.out.println(integerExample.getClass()); // class GenericExample

        // 因此,在运行时无法获取泛型的具体类型
        System.out.println(stringExample.getData()); // Hello
        System.out.println(integerExample.getData()); // 123

        // 编译器插入了类型转换以确保类型安全
        // 运行时会执行类型转换
        String str = stringExample.getData(); // No explicit casting needed
        Integer num = integerExample.getData(); // Explicit casting needed
    }
}

        在上面的例子中,尽管我们在实例化 GenericExample 时指定了具体的类型参数,但在运行时,通过 getClass() 方法可以看到实际的类型是 GenericExample,而不是 GenericExample<String> 或 GenericExample<Integer>。这就是类型擦除的表现。在获取数据时,编译器会自动插入类型转换以确保类型安全。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/499191.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Qt+OpenGL入门教程(二)——OpenGL渲染管线

渲染管线是图形学不可或缺的&#xff0c;在学习它之前&#xff0c;我们先了解一下什么是管线&#xff1f; 管线/流水线 当我们谈到管线时&#xff0c;我们指的是一个由多个阶段组成的过程&#xff0c;每个阶段都完成任务的一部分。在现实世界中&#xff0c;流水线的概念在许多…

小白了解Pinia第2集 · 三大核心状态Getters、Actions以及Plugins 插件

三大核心状态 state 第1集有详细讲解&#xff1a;https://blog.csdn.net/qq_51463650/article/details/137137080?spm1001.2014.3001.5501 getters Getter 完全等同于 Store 状态的 计算值。 它们可以用 defineStore() 中的 getters 属性定义。 他们接收“状态”作为第一个…

C++从入门到精通——缺省参数

缺省参数 前言一、缺省参数概念二、缺省参数分类位置参数的缺省参数全缺省参数半缺省参数 关键字参数的缺省参数函数指针的缺省参数lambda表达式 三、缺省参数的具体代码展示main.cpp 前言 缺省参数是在函数定义时指定的默认值&#xff0c;当调用函数时未提供该参数的值时&…

统信 UOS V20 一键安装 Oracle 12CR2(220118)单机版

Oracle 一键安装脚本&#xff0c;演示 统信 UOS V20 一键安装 Oracle 12CR2&#xff08;220118&#xff09;单机版过程&#xff08;全程无需人工干预&#xff09;&#xff1a;&#xff08;脚本包括 ORALCE PSU/OJVM 等补丁自动安装&#xff09; ⭐️ 脚本下载地址&#xff1a;…

云主机8核16G配置租用优惠价格1198元1年、4688元三年

京东云8核16G租用优惠价格1198元1年、4688元三年&#xff0c;配置为8C16G-270G SSD系统盘-5M带宽-500G月流量&#xff0c;华北-北京地域。京东云8核16G服务器活动页面 atengyun.com/go/jd 京东云8核16G租用优惠价格 京东云&#xff1a;轻量云主机CPU内存&#xff1a;8C16G公网带…

AIGC重塑金融 | 大模型在金融行业的应用场景和落地路径

作者&#xff1a;林建明 来源&#xff1a;IT阅读排行榜 本文摘编自《AIGC重塑金融&#xff1a;AI大模型驱动的金融变革与实践》&#xff0c;机械工业出版社出版 目录 01 大模型在金融领域的 5 个典型应用场景 02 大模型在金融领域应用所面临的风险及其防范 03 AIGC 技术的科…

职场沟通教训 程序汪改了一行代码,导致测试和开发大战

本文章有视频的&#xff0c;请到B站 我是程序汪 观看 程序汪改了一行代码&#xff0c;导致测试和开发大战&#xff0c;职场沟通教训 程序汪改了一行代码&#xff0c;导致测试和开发大战 鸡汤文 每个人都会在沟通上出问题 工作上沟通出问题可能让你郁闷一天、丢了客户、损失金…

解决Nginx请求转发将POST变为GET的问题

先说问题 我配置了Nginx代理&#xff0c;目的是将请求转发到指定的后端&#xff0c;对于普通的JSON数据&#xff0c;没有什么问题。 但是有文件上传的就不一样了&#xff0c;我需要指定到第3方的地址。然而常规的配置完成后,Nginx实际的转发动作改变了我的请求方式 location …

鸿蒙开发人才紧缺!这份《HarmonyOS教学视频》帮你更快上手鸿蒙

去年9月&#xff0c;华为宣布鸿蒙原生应用全面启动&#xff0c;基于开源鸿蒙开发的 HarmonyOS NEXT 鸿蒙星河版将在今年秋天正式和消费者见面。该版本系统底座将由华为全线自研&#xff0c;去掉传统安卓 AOSP 代码。 这意味着&#xff0c;鸿蒙星河版将不再兼容安卓应用&#xf…

系统分析师-参考模型

前言 网络术语中的参考模型指的是OSI参考模型&#xff0c;由ISO&#xff08;国际标准化组织&#xff09;制定的一套普遍适用的规范集合&#xff0c;以使得全球范围的计算机平台可进行开放式通信。 ISO创建了一个有助于开发和理解计算机的通信模型&#xff0c;即开放系统互联OS…

SQL复习专题

请结合B站-技术蛋老师 视频学习 核心语法 一、增&#xff1a;数据库/表格 create create database 数据库名&#xff1b;#创建表&#xff08;列名类型&#xff09; mysql> create table eggs_record(-> id int,-> egg_name varchar(10),-> sold date-> ); 这…

基于单片机控制的智能轿车停车场设计

**单片机设计介绍&#xff0c;基于单片机控制的智能轿车停车场设计 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于单片机控制的智能轿车停车场设计是一个集成了现代电子技术、自动化技术、计算机技术的综合性项目。该设计旨…

com.alibaba.boot.nacos.config.binder.NacosBootConfigurationPropertiesBinder解决记录

一直正常的服务突然启动失败了&#xff0c;控制台报错 查询后发现是spring-boot-starter版本2.4和nacos-config 0.2.8版本冲突了 于是看了下nacos-config版本&#xff0c;发现有两个如下 但是原来启动正常&#xff0c;看了下老版本代码发现nacos-config-springboot-autoconfig…

亚马逊跨境电商迎来崭新时代,武汉星起航携手卖家共赴新征程

随着全球经济一体化的深入发展&#xff0c;跨境电商已成为推动国际贸易的重要力量。据最新数据显示&#xff0c;2023年中国跨境电商出口规模达到1.83万亿元&#xff0c;同比增长19.6%&#xff0c;增速远超电商行业整体水平。在这一背景下&#xff0c;2024年有望成为中国跨境电商…

CSGO赛事管理系统的设计与实现|Springboot+ Mysql+Java+ B/S结构(可运行源码+数据库+设计文档)

本项目包含可运行源码数据库LW&#xff0c;文末可获取本项目的所有资料。 推荐阅读100套最新项目持续更新中..... 2024年计算机毕业论文&#xff08;设计&#xff09;学生选题参考合集推荐收藏&#xff08;包含Springboot、jsp、ssmvue等技术项目合集&#xff09; 目录 1. 系…

蓝桥 python笔记12——动态规划、二维dp、最长上升子序列、最长公共子序列

目录 动态规划 二维dp 最长上升子序列 最长公共子序列 动态规划 # dp[n]表示n个台阶方案数 # dp[n]dp[n-1]dp[n-2] # dp[1]1 dp[2]2 nint(input()) dp[0]*(n1) dp[1]1 dp[2]2 for i in range(3,n1):dp[i]dp[i-1]dp[i-2] print(dp[n])二维dp Nint(input()) a[[0]*(N1)]# 下…

【python】爬取4K壁纸保存到本地文件夹【附源码】

欢迎来到英杰社区https://bbs.csdn.net/topics/617804998 图片信息丰富多彩&#xff0c;许多网站上都有大量精美的图片资源。有时候我们可能需要批量下载这些图片&#xff0c;而手动一个个下载显然效率太低。因此&#xff0c;编写一个简单的网站图片爬取程序可以帮助我们…

开通幻兽帕鲁游戏多人联机服务器多少钱?价格意想不到

2024年全网最全的幻兽帕鲁服务器租用价格表&#xff0c;阿里云幻兽帕鲁游戏服务器26元1个月、腾讯云32元一个月、京东云26元一个月、华为云24元1个月&#xff0c;阿腾云atengyun.com整理最新幻兽帕鲁专用4核16G、8核16G、8核32G游戏服务器租用价格表大全&#xff1a; 阿里云幻…

TSINGSEE青犀多模型、算力调度与智能分析AI算法中台介绍及应用

TSINGSEE青犀AI算法中台是一款平台型产品&#xff0c;专注于提供各行业中小场景中部署解决方案。平台具备接入广、性能强、支持跨平台、芯片国产化等特点&#xff0c;可提供丰富的视图接入能力和智能分析能力。平台将不同类型、不同协议前端设备&#xff0c;支持通过不同网络环…

动手学机器学习线性回归+习题

线性回归 矩阵求导&#xff1a; 左边是分子布局&#xff0c;右边是分母布局&#xff0c;一般都用分母布局 解析解与数值解&#xff1a; 解析解是严格按照公式逻辑推导得到的&#xff0c;具有基本的函数形式。给出任意的自变量就可以求出其因变量 数值解是采用某种计算方法&a…