数据结构:包装类和泛型

server/2025/1/8 19:09:53/

目录

一、包装类

1、基本数据类型和对应的包装类

 2、装箱和拆箱 

3、自动装箱和自动拆箱 

二、泛型 

1、什么是泛型

2、泛型语法 

3、泛型类 

4、擦除机制 

5、泛型的上界 

6、泛型方法

三、通配符 

1、什么是通配符 

2、通配符上界 

3、通配符下界 


📚一、包装类

包装类是对应着各种基本数据类型进行包装后产生的引用数据类型,我们甚至可以称它们为基本数据类型的plus版本。

至于我们为什么要设计包装类这是因为Java是一个面向对象的编程语言,但是Java中的八种基本数据类型却是不面向对象的,为了使用方便和解决这个不足,在设计类时为每个基本数据类型设计了一个对应的类进行代表,这样八种基本数据类型对应的类统称为包装类

🐬1、基本数据类型和对应的包装类

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

🐬 2、装箱和拆箱 

java"> int i = 10;
//装箱操作,新建⼀个Integer 类型对象,将i 的值放⼊对象的某个属性中Integer ii = Integer.valueOf(i);Integer ij = new Integer(i);//拆箱操作,将Integer 对象中的值取出,放到⼀个基本数据类型中int j = ii.intValue();

然而我们发现在使用过程中这种装箱和拆箱的过程,带了不少的代码量,于是为了减少开发时的负担,java提供了自动装箱和拆箱机制。

🐬3、自动装箱和自动拆箱 

java">public class  Test{public static void main(String[] args) {int i = 10;Integer ii = i;//⾃动装箱Integer ij = (Integer)i;//⾃动装箱int j = ii;//⾃动拆箱int k = (int)ii;//⾃动拆箱}
}

我们可以发现这种自动的装箱和拆箱方式大大的减少了我们的代码量和开发负担 

现在我们已经大致了解了包装类了,接下来让我们来看一道题

java">public class  Test{public static void main(String[] args) {Integer a = 127;Integer b = 127;Integer c = 128;Integer d = 128;System.out.println(a == b);System.out.println(c == d);}
}

他的输出结果是什么呢?可能我们下意识认为都是true,但我们看一下运行后给出的结果 

我们发现并不都是true这是为什么呢?

首先我们先来了解一下“==”“==”在比较基本数据类型的时候,就是字面值的比较,而现在我们并不是基本数据类型,而是由基本类型转变的包装类属于引用类型,而“==” 在比较引用类型的时候,比较的是地址,但是由于Integer是具有范围的-128~127,在这区间的值是被存放在常量池中,所以只有是处于这个区间的相同值进行比较都是相同的,但是128超出了这个范围,因此两个变量的值虽然都等于128,但并不是同一个地址,因此不为true。

📚二、泛型 

🐬1、什么是泛型

一般的类和方法,只能使用具体的类型:要么是基本类型,要么是自定义的类。如果要编写可以应用于多种类型的代码,这种刻板的限制对代码的束缚就会很大。

因此我们在JDK1.5引入的新的语法:泛型,所谓泛型就是适用于许多许多类型。从代码上讲,就是对类型实现了参数化

我们来看这样一个例子,在数组中存放各种类型的元素

我们知道Object类默认为所有类的父类,所以我们创建的数组为Object类的数组。

创建完后我们发现的确任何的类型我们都能够进行存放了,但是当我们进行元素的取出时我们发现,编译器进行了报错,这时因为我们要取出的虽然本身是字符串,但是我们必须进行强制类型转换。

但是,更多情况下,我们还是希望他只能够持有一种数据类型。而不是同时持有这么多类型。所以,泛型的主要目的:就是指定当前的容器,要持有什么类型的对象,让编译器去做检查。 

🐬2、泛型语法 

基础写法:

java">class 泛型类名称 < 类型形参列表 > {// 这里可以使用类型参数}class ClassName<T1,T2,T3,...,Tn>{}

其他写法: 

java">class 泛型类名称<类型形参列表> extends 继承类/* 这⾥可以使⽤类型参数 */ { // 这⾥可以使⽤类型参数
}class ClassName<T1, T2, ..., Tn> extends ParentClass<T1> {// 可以只使⽤部分类型参数
}

因此我们就可以将上述代码写成新的样子 

java">class MyArray1<T> {public Object[] array = new Object[10];//public T[] array = (T[]) new Object[10];两种写法都可以public T getPos(int pos) {return (T)this.array[pos];}public void setVal(int pos,T val) {this.array[pos] = val;}
}
public class TestDemo1 {public static void main(String[] args) {MyArray1<Integer> myArray1 = new MyArray1<>();myArray1.setVal(0,10);int ret1 = myArray1.getPos(0);System.out.println(ret1);MyArray1<String> myArray2 = new MyArray1<>();myArray2.setVal(0,"hello");String ret2 = myArray2.getPos(0);System.out.println(ret2);}
}

 

我们发现利用这种方法我们就可以吧不同类型的相同操作只利用一段代码并在主函数中传入不同的类型就可全部使用,这就是我们使用泛型的原因。

当然泛型还是有一些规范的

  1. 类名后的<T>代表占位符,表示当前类是一个泛型
  2. 类型形参一般使用一个大写字母表示,常用的名称有:
  • E表示Element
  • K表示Key
  • V表示Value 
  • N表示Number
  • T表示Type
  • S,U,V等等-第二、第三、第四个类型

🐬3、泛型类 

java">泛型类<类型实参> 变量名;        // 定义⼀个泛型类引⽤new 泛型类<类型实参>(构造⽅法实参);        //实例化⼀个泛型类对象MyArray<Integer> list = new MyArray<Integer>();当编译器可以根据上下⽂推导出类型实参时,可以省略类型实参的填写MyArray<Integer> list = new MyArray<>();

注意:泛型只能接受类,所有的基本数据类型必须使用包装类! 

🐬4、擦除机制 

现在我们已经大致了解了泛型的用法了,但是大家有没有想过泛型是怎么在编译器上进行编译的呢,而这就涉及到了我们的擦除机制

  1. 在编译时,Java编译器会将泛型类型信息从代码中移除,这个过程就叫做类型擦除。
  2. 擦除后,泛型类型会被替换为其边界类型(通常是Object)或者指定的类型。

擦除过程

  1. 泛型参数替换为其边界或Object
  2. 在必要的地方插入类型转换以保持类型安全。
  3. 生成桥接方法以保持多态性。 

 

🐬5、泛型的上界 

在定义泛型类时,有时需要对传入的类型变量做一定的约束,可以通过类型边界来约束。

java">class 泛型类名称<类型形参 extends 类型边界> {...
}
public class MyArray<E extends Number> {...
}

而类型边界中所谓的上界,其实就是代表我们规定的边界作为父类,而我传入的变量必须是父类本身或者是父类的子类

例如Number的子类有Integer,没有String

java">public class MyArray<E extends Number> {...
}MyArray<Integer> l1;//正常,因为Integer是Number的⼦类型MyArray<String> l2;//编译错误,因为String不是Number的⼦类型

对于接口也是一样的

java"> // E必须是实现了Comparable接⼝的
public class MyArray<E extends Comparable<E>> {...}

注意:没有指定类型边界E,可以视为EextendsObject 

🐬6、泛型方法

java">⽅法限定符 <类型形参列表> 返回值类型 ⽅法名称(形参列表) { ... }

 例子:交换元素

📚三、通配符 

🐬1、什么是通配符 

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

我们发现相同类型的引用是可以相互指向的,而不同类型的引用是不能相互指向的,那么我们可不可建立一个能够指向多种不同泛型参数的对象呢?这就需要用到我们的通配符了 

我们发现当我们使用了通配符?之后就可以指向不同的泛型参数了 。

当然,有些时候我们同样希望它能够像泛型一样只指向像一部分的泛型参数,所以这时我们在通配符上也引用了上界,同时通配符多了一项对下界的限制

? extends 类:设置通配符上限

? super 类:设置通配符下限

🐬2、通配符上界 

java"> <? extends 上界> 
<? extends Number>//可以传⼊的实参类型Number或者Number的⼦类

 跟泛型的上界一样,同样将通配符限制为界限的本身和他的子类

🐬3、通配符下界 

java"> <? super 下界> 
<? super Integer>//代表可以传⼊的实参的类型是Integer或者Integer的⽗类类型

而下界则是,通配符的范围只能界限的本身和他的父类


好了今天的分享就到这里,还请大家多多关注,我们下一篇见! 


http://www.ppmy.cn/server/156859.html

相关文章

基于CentOS的Docker + Nginx + Gitee + Jenkins部署总结(进阶)-- 接入钉钉通知功能

前言 在实际项目会出现更多复杂需求&#xff0c;如一个项目多个端&#xff08;admin、h5等&#xff09;、多分支情况&#xff08;dev、其他分支&#xff09;、多接口环境&#xff08;dev/prod/test等&#xff09;、是否需要钉钉通知等个性化功能。 一、 参数化构建配置 在基础…

30天开发操作系统 第 12 天 -- 定时器

前言 定时器(Timer)对于操作系统非常重要。它在原理上却很简单&#xff0c;只是每隔一段时间(比如0.01秒)就发送一个中断信号给CPU。幸亏有了定时器&#xff0c;CPU才不用辛苦地去计量时间。……如果没有定时器会怎么样呢?让我们想象一下吧。 假如CPU看不到定时器而仍想计量时…

密码学复习

目录 密码体系相关概念传统密码替换技术单表替换密码双表替换密码playfair密码多表替换密码维基密亚密码 置换技术栅栏密码列移位密码 对称密码&#xff08;block cipher&#xff09;DES工作模式AES近现代的一些对称密钥Trible-DESIDEAblowfish 对称密码&#xff08;stream cip…

【计算机组成原理课程设计】:实验0 ROM仿真、实验1 验证74L181运算和逻辑功能、实验2 运算器 2、实验 3 跑马灯、实验4 模拟微程序实现指令

下面文件都放在 gitee 中&#xff0c;大家可以自行选择拿取 course_design: 课程设计 - Gitee.comhttps://gitee.com/island0920/course_design/tree/master/%E8%AE%A1%E7%BB%84 前言 -- 如何使用 Multisim 1. 如何使用元器件 2. 常用元器件 VCC 接地 key space &#xf…

若依中Feign调用的具体使用(若依微服务版自身已集成openfeign依赖,并在此基础上定义了自己的注解)

若依中Feign调用具体使用 注意&#xff1a;以下所有步骤实现的前提是需要在启动类上加入注解 EnableRyFeignClients 主要是为开启feign接口扫描 1.创建服务提供者(provider) 导入依赖(我在分析依赖时发现若依本身已经引入openfeign依赖,并在此基础上自定义了自己的EnableRyF…

【微服务】5、服务保护 Sentinel

Sentinel学习内容概述 Sentinel简介与结构 Sentinel是Spring Cloud Alibaba的组件&#xff0c;由阿里巴巴开源&#xff0c;用于服务流量控制和保护。其内部核心库&#xff08;客户端&#xff09;包含限流、熔断等功能&#xff0c;微服务引入该库后只需配置规则。规则配置方式有…

【wiki知识库】08.添加用户登录功能--后端SpringBoot部分

目录 一、今日目标? 二、SpringBoot后端实现 2.1 新增UserLoginParam 2.2 修改UserController 2.3 UserServiceImpl代码 2.4 创建用户上下文工具类 2.5?通过token校验用户&#xff08;重要&#xff09; 2.6 创建WebMvcConfig 2.7 用户权限校验拦截器 一、今日目标 上…

c++程序设计(第3版)系列教程

c程序设计(第3版)系列笔记 预备知识 在c当中&#xff0c;避免字符串被截断的输入为gets(S)&#xff0c;但是由于c语言新标准的推行和部分删除&#xff0c;在使用gets(S)时只能通过宏定义#define gets(S) fgets(S,sizeof(S),stdin)处理之后使用。 在c当中&#xff0c;面对难以处…