数据结构(Java版)第二期:包装类和泛型

       

目录

一、包装类

1.1. 基本类型和对应的包装类

1.2. 装箱和拆箱

1.3. 自动装箱和自动拆箱

二、泛型的概念

三、引出泛型

3.1. 语法规则

3.2. 泛型的优点

四、类型擦除 

4.1. 擦除的机制 

五、泛型的上界

5.1. 泛型的上界的定义

5.2. 语法规则 

六、泛型方法

6.1. 定义语法 

6.2. 交换方法的实例

七、通配符


包装类和泛型我们在Java语法中,我们在基本数据类型里面涉及过,但是我们在语法里面用不到,而在数据结构里面我们才会有应用的。 

一、包装类

1.1. 基本类型和对应的包装类

         Java共有8种基本数据类型,Java给这些基本类型都搞了一个类进行表示,来对这些类进行一个封装,这就是包装类。 

基本类型包装类
byteByte
shortShort
intInteger
longLong
floatFloat
doubleDouble
charCharacter
booleanBoolean

1.2. 装箱和拆箱

        在Java当中,提供了一些操作,使包装类和内置类型可以相互转换。内置类型转为包装类型称为装箱,包装类型转为内置类型称为拆箱。但这些代码写法已经过时了,我们需要重点掌握的是⾃动装箱和⾃动拆箱。

1.3. 自动装箱和自动拆箱

public class Main {
    public static void main(String[] args) {
        int i = 10;
        Integer ii = i;//自动装箱
        Integer ij = (Integer) i;//自动装箱,后面的(Integer)可有可无

        int j = ii;//自动拆箱
        int k = (int)ii;//自动拆箱,后面的(int)可有可无
    }
}

       我们可以通过javap-c查看字节码⽂件内容,观察装箱和拆箱的操作。我们可以在IDEA里面装一个jclasslib ByteCode Viewer的插件,然后点击View,再点击Show Bytecode With Jclasslib。我们点到main方法,点击code,就可以看到所对应的字节码文件。

 

         我们来看下面的一段代码,此时的a,b,c,d,e都是引用类型变量。当赋值相同时,结果就是true;当赋值不同时,结果就是false。当赋值超出包装类型的范围时,无论赋值相不相等,结果都是false。 

       这是因为Integer里面的常量值放在常量池当中,我们进行赋值,相当于在常量池中进行取值,如果超出这个值,那么就是池内与池外的进行比较,结果就是false。 

public class Main {
    public static void main(String[] args) {
        Integer a = 127;
        Integer b = 127;
        System.out.println(a == b);//true

        Integer c = 128;
        Integer d = 128;
        System.out.println(c == d);//false

        Integer e = 126;
        System.out.println(a == e);//false
    }
}

二、泛型的概念

       很多编程语言都有泛型这样的语法机制。在Java中,写一个类或者是方法,需要声明方法里面的成员或者参数的类型。但也有些情况下,需要一个类或者方法能够多种类型支持。也就是一份代码,支持多种数据类型。

三、引出泛型

3.1. 语法规则

class 泛型名称<参数列表>{
      //这里可以使用类型参数;
}

class ClassName<T1,T2,……,Tn>{
}

        参数列表中要把类中会用的哪些类型列出来,后续使用这个类,创建实例的时候也要同时指定泛型参数的实参。T1,T2相当于类型的形参。

private T[] arrays = new T[];
//这种写法是错误的

        因为T要表示任何类型,new T[]的时候就可能会涉及到该类的构造方法,T是什么类型不知道该怎么办?就得先写成Object[]再进行强转。

T[] arrays = (T[]) new Object[];
class MyArray<T>{
    T[] arrays = (T[]) new Object[10];
    public T get(int index){
        return arrays[index];//获取数组的下标
    }
    public void set(int index,T value){
        arrays[index] = value;//对数组进行赋值
    }
}
//上面的T不用再进行强转了

//对方法的实现
public class Main {
    public static void main(String[] args) {
        MyArray<String> array1 = new MyArray<String>();//里面可以存放字符。代码是灰色的,表示可以不写
        MyArray<Integer> array2 = new MyArray<>();//里面可以存放整数
        MyArray array4 = new MyArray();//裸类型,这种写法是不科学的
        MyArray<int> array3 = new MyArray<>();//error: Type argument cannot be of primitive type
    }
}

3.2. 泛型的优点

       1.代码重用,一份代码,支持多种类型; 2.自动地进行类型转化,编译过程中会自动触发一些类型检查。

四、类型擦除 

4.1. 擦除的机制 

      Java的泛型,本质上是通过Object类进行编译的。编译器生成代码的时候,自动进行类型转化。比如下面的代码中的get方法,我要对T转化成String类型,编译器从数组中拿到的是一个Object类,然后进行自动转化成String类,返回到调用位置。在set方法里面,set String进来,编译器再自动把String转化成Object。

class MyArray<T>{
    T[] arrays = (T[]) new Object[10];
    public T get(int index){
        return arrays[index];
    }
    public void set(int index,T value){
        arrays[index] = value;
    }
}

 下面是一段擦除的代码用例

//擦除前
class MyArray<T>{
    public Object[] arrays = new Object[10];
    public T getPos(int pos){
        return (T)this.arrays[pos];
    }
    public void setVal(int pos,T val){
        this.arrays[pos] = val;
    }
}
//擦除后
class MyArray<T>{
    public Object[] arrays = new Object[10];
    public Object getPos(int pos){
        return this.arrays[pos];
    }
    public void setVal(int pos,T val){
        this.arrays[pos] = val;
    }
}

五、泛型的上界

5.1. 泛型的上界的定义

     描述的是使用泛型,创建泛型实例的时候,传入的参数(类型实参)需要满足什么条件。 

5.2. 语法规则 

class 泛型名称<类型实参 extends 类型边界>{
}

       这个类型边界相当于是“父类”,后续创建类型实例的类型参数,必须是这个父类的子类。比如我们要写一个算术运算的泛型类,泛型参数必须给数字。

class MyArray<E extends Number>{
     MyArray<Integer> l1;//正常
     MyArray<String> l2;//错误
}

     如果没有指定类型边界E,可以视为E extends Object。

六、泛型方法

6.1. 定义语法 

方法限定符 <类型形参列表> 返回值类型 方法名称{
}

6.2. 交换方法的实例

public class Main {
    //静态的泛型方法需要在static后面用<>声明泛型类型参数
    public static <E> void swap(E[] array,int i,int j){
        E t = array[i];
        array[i] = array[j];
        array[j] = t;
    }
}

七、通配符

     前面的知识都是在定义泛型时涉及到的,通配符是针对泛型实例化的时候涉及到的。

class MyClass<T>{

}
public class Main {
    public static void main(String[] args) {
        MyClass<Integer> obj1 = new MyClass<>();
        MyClass<String> obj2 = new MyClass<>();
        MyClass<Integer> obj3 = new MyClass<>();

        obj1 = obj3;//正常
        obj1 = obj2;//错误
    }
}

     因为Obj1与Obj2类型不相同,所以会报错。那我们能否创建一种引用,能够指向多种泛型参数的对象呢?这时就要用到通配符了。

MyClass<?> obj4 = obj3; 
MyClass<? extends Number> obj5 = obj1;
obj5 = obj2;
//这个代码不符合要求,约定obj5的通配符,只能匹配到Number和它的子类;
//因为通配符只能在泛型实例化时使用

       这里的代码不要和泛型的上界搞混。我们除了可以指定父类,还能指定子类。

MyClass<? super Integer> obj6 = obj1;//此处的通配符只能匹配到Integer和它的父类
obj6 = new MyClass<double>();//double并不是Integer的子类,所以会报错
obj6 = new MyClass<Number>();//Number是Integer的父类。

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

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

相关文章

ffmpeg 视频滤镜:高斯模糊-gblur

滤镜描述 gblur 官网地址 > FFmpeg Filters Documentation 这个滤镜会将视频变得模糊。 滤镜使用 参数 gblur AVOptions:sigma <float> ..FV.....T. set sigma (from 0 to 1024) (default 0.5)steps <int> ..FV.....T…

vue中路由缓存

vue中路由缓存 问题描述及截图解决思路关键代码及打印信息截图 问题描述及截图 在使用某一平台时发现当列表页码切换后点击某一卡片进入详情页后&#xff0c;再返回列表页时页面刷新了。这样用户每次看完详情回到列表页都得再重新输入自己的查询条件&#xff0c;或者切换分页到…

eclipse-git项目提示NO-HEAD

1、出现该问题的过程 本人在用eclipse拉取git代码&#xff0c;刚拉取完&#xff0c;可能还没来得及跟本地的分支合并&#xff0c;电脑就卡动了。无奈只能重启电脑&#xff0c;打开eclipse&#xff0c;maven项目后面就出现了xxx NO-HEAD的提示。 2、问题解决 根据错误提示&am…

网络安全与加密

1.Base64简单说明描述&#xff1a;Base64可以成为密码学的基石&#xff0c;非常重要。特点&#xff1a;可以将任意的二进制数据进行Base64编码结果&#xff1a;所有的数据都能被编码为并只用65个字符就能表示的文本文件。65字符&#xff1a;A~Z a~z 0~9 / 对文件进行base64编码…

goframe开发一个企业网站 在vue-next-admin 显示验证码 19

index.go 文件中的代码&#xff0c;我将为该文件中的主要功能和方法添加注释&#xff0c;并生成一篇 Markdown 格式的文章。这将包括对每个函数的用途、输入参数和返回值的简要说明。 index.go 包和导入 package adminimport ("context""errors""gf…

数据库的联合查询

数据库的联合查询 简介为什么要使⽤联合查询多表联合查询时MYSQL内部是如何进⾏计算的构造练习案例数据案例&#xff1a;⼀个完整的联合查询的过程 内连接语法⽰例 外连接语法 ⽰例⾃连接应⽤场景示例表连接练习 ⼦查询语法单⾏⼦查询多⾏⼦查询多列⼦查询在from⼦句中使⽤⼦查…

Oracle 23ai 对应windows版本安装配置PLSQL导入pde文件navicat连接Oracle

因为有一个pde文件需要查看里面的数据&#xff0c;所以这次需要配置本地oracle数据库&#xff0c;并且导入数据&#xff0c;因为还有navicat&#xff0c;所以就想用navicat去连接查看。 1、找到官网。 Get Started with Oracle Database 23ai | Oracle 2、下载windows版本。…

Juc01_多线程概述、四种实现方式、常用方法API、生命周期、买票案例、synchronized锁

目录 本章讲述内容&#xff1a;多线程概述、四种实现方式、常用方法API、生命周期、买票案例、synchronized锁 ①. 多线程的概述 ②. 多线程的实现方式 ①. 继承Thread ②. 实现Runnable接口 ③. Callable接口(创建线程) ④. 线程池 ③. 设置和获取线程名称 ④. 线程…

一个高度可扩展的 Golang ORM 库【GORM】

GORM 是一个功能强大的 Golang 对象关系映射&#xff08;ORM&#xff09;库&#xff0c;它提供了简洁的接口和全面的功能&#xff0c;帮助开发者更方便地操作数据库。 1. 完整的 ORM 功能 • 支持常见的关系模型&#xff1a; • Has One&#xff08;一对一&#xff09; • …

ubuntu24挂载硬盘记录

1、显示硬盘及所属分区情况。在终端窗口中输入如下命令&#xff1a; sudo fdisk -l 找到自己硬盘的分区 我的地址/dev/sda 2、显示硬盘及所属分区情况。在终端窗口中输入如下命令&#xff0c;格式化自己硬盘&#xff1a; sudo mkfs -t ext4 /dev/sda 3、在终端窗口中输入如下…

Flink四大基石之Window

为什么要用WIndow 在流处理应用中&#xff0c;数据是连续不断的&#xff0c;有时我们需要做一些聚合类的处理&#xff0c;例如&#xff1a;在过去的1分钟内有多少用户点击了我们的网页。 在这种情况下&#xff0c;我们必须定义一个窗口(window)&#xff0c;用来收集最近1分钟内…

使用ENSP实现默认路由

一、项目拓扑 二、项目实现 1.路由器AR1配置 进入系统试图 sys将路由器命名为R1 sysname R1关闭信息中心 undo info-center enable 进入g0/0/0接口 int g0/0/0将g0/0/0接口IP地址配置为2.2.2.1/24 ip address 2.2.2.1 24进入g0/0/1接口 int g0/0/1将g0/0/1接口IP地址配置为1.…

《基于FPGA的便携式PWM方波信号发生器》论文分析(三)——数码管稳定显示与系统调试

一、论文概述 基于FPGA的便携式PWM方波信号发生器是一篇由任青颖、庹忠曜、黄洵桢、李智禺和张贤宇 等人发表的一篇期刊论文。该论文主要研究了一种新型的信号发生器&#xff0c;旨在解决传统PWM信号发生器在移动设备信号调控中存在的精准度低和便携性差的问题 。其基于现场可编…

vue 预览pdf 【@sunsetglow/vue-pdf-viewer】开箱即用,无需开发

sunsetglow/vue-pdf-viewer 开箱即用的pdf插件sunsetglow/vue-pdf-viewer, vue3 版本 无需多余开发&#xff0c;操作简单&#xff0c;支持大文件 pdf 滚动加载&#xff0c;缩放&#xff0c;左侧导航&#xff0c;下载&#xff0c;页码&#xff0c;打印&#xff0c;文本复制&…

1-golang_org_x_crypto_bcrypt测试 --go开源库测试

1.实例测试 package mainimport ("fmt""golang.org/x/crypto/bcrypt" )func main() {password : []byte("mysecretpassword")hashedPassword, err : bcrypt.GenerateFromPassword(password, bcrypt.DefaultCost)if err ! nil {fmt.Println(err)…

嵌入式的C/C++:深入理解 static、const 与 volatile 的用法与特点

目录 一、static 1、static 修饰局部变量 2、 static 修饰全局变量 3、static 修饰函数 4、static 修饰类成员 5、小结 二、const 1、const 修饰普通变量 2、const 修饰指针 3、const 修饰函数参数 4. const 修饰函数返回值 5. const 修饰类成员 6. const 与 #defi…

超高流量多级缓存架构设计!

文章内容已经收录在《面试进阶之路》&#xff0c;从原理出发&#xff0c;直击面试难点&#xff0c;实现更高维度的降维打击&#xff01; 文章目录 电商-多级缓存架构设计多级缓存架构介绍多级缓存请求流程负载均衡算法的选择轮询负载均衡一致性哈希负载均衡算法选择 应用层 Ngi…

【C++ 算法进阶】算法提升二十三

目录 左右数组相减绝对值最大值 &#xff08;题意代换&#xff09;题目题目分析 可整合数组 &#xff08;题意代换&#xff09;题目题目分析代码 水王问题题目题目分析代码水王问题变形思路讲解 合并石头的最低成本 &#xff08;动态规划&#xff09;题目题目分析代码 左右数组…

solr 远程命令执行 (CVE-2019-17558)

漏洞描述 Apache Velocity是一个基于Java的模板引擎&#xff0c;它提供了一个模板语言去引用由Java代码定义的对象。Velocity是Apache基金会旗下的一个开源软件项目&#xff0c;旨在确保Web应用程序在表示层和业务逻辑层之间的隔离&#xff08;即MVC设计模式&#xff09;。 Apa…

idea怎么打开两个窗口,运行两个项目

今天在开发项目的时候&#xff0c;前端希望运行一下以前的项目&#xff0c;于是就需要开两个 idea 窗口&#xff0c;运行两个项目 这里记录一下如何设置&#xff1a;首先依次点击&#xff1a; File -> Settings -> Appearance & Behavior ->System Settings 看到如…