【Java数据结构 -- 包装类和泛型】

包装类和泛型

  • 1. 包装类
    • 1.1 基本数据类型和对应的包装类
    • 1.2 装箱和拆箱
    • 1.3 自动装箱和自动拆箱
    • 1.4 自动装箱实际上是调用了valueOf()
    • 1.5 Integer包装类赋值注意点
  • 2 什么是泛型
  • 3 引出泛型
  • 4 泛型的使用
    • 4.1 语法
    • 4.2 类型推导
  • 5 裸类型
  • 6 泛型如何编译
    • 6.1 擦除机制
  • 7 泛型的上界
    • 写一个泛型类, 求一个数组当中的最大值
    • 另一个类作为参数实例化时要引用它对应的接口和重写对应的方法
  • 8 泛型方法
    • 设为静态方法static

1. 包装类

在Java中,由于基本类型不是继承自Object,为了在泛型代码中可以支持基本类型,Java给每个基本类型都对应了一个包装类型。

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

基本数据类型 --> 包装类
byte --> Byte
short --> Short
int --> Integer
long --> Long
float --> Float
double --> Double
char --> Character
boolean --> Boolean
除了 Integer 和 Character, 其余基本类型的包装类都是首字母大写。

1.2 装箱和拆箱

装箱/装包:把一个基本类型转变为包装类型
拆箱/拆包:把一个包装类型转变为基本类型

1.3 自动装箱和自动拆箱

    public static void main(String[] args) {int a = 10;Integer ii = a; //自动装箱Integer ii2 = new Integer(10);int b = ii2; //自动拆箱System.out.println(ii);  // 10System.out.println(b);   // 10}

1.4 自动装箱实际上是调用了valueOf()

    public static void main2(String[] args) {int a = 10;//Integer ii = a; //自动装箱Integer ii = Integer.valueOf(a);  // 手动装箱Integer ii2 = new Integer(10);//int b = ii2; //自动拆箱int b = ii2.intValue(); // 手动拆箱double d = ii2.intValue();System.out.println(ii);  // 10System.out.println(b);   // 10System.out.println(d);   // 10.0}

1.5 Integer包装类赋值注意点

    public static void main(String[] args) {//  i >= IntegerCache.low && i <= IntegerCache.high// 在 Integer  valueOf方法中  的int i取值范围是  -128~127// 在这个返回返回的是  -128 到 127 中 255 个地址 存放100// IntegerCache.cache[i + (-IntegerCache.low)] 即返回的地址为同一个地址// 而200不属于这个取值范围里面 返回的是 一个新的 值 new Integer(i)/*Integer ii = 100;Integer ii2 = 100;  //true */Integer ii = 200;Integer ii2 = 200;  //falseSystem.out.println(ii == ii2);}

2 什么是泛型

一般的类和方法,只能使用具体的类型: 要么是基本类型,要么是自定义的类。如果要编写可以应用于多种类型的代码,这种刻板的限制对代码的束缚就会很大。
泛型是在JDK1.5引入的新的语法,通俗讲,泛型:就是适用于许多许多类型。从代码上讲,就是对类型实现了参数化

3 引出泛型

实现一个类,类中包含一个数组成员,使得数组中可以存放任何类型的数据,也可以根据成员方法返回数组中某个下标的值?

class MyArray <T>{// <T> 占位符 是一个泛型类//  泛型的意义:1. 在编译的时候 检查数据类型是否正确 2.在编译的时候   帮助进行类型转化//public Object[] array = new Object[10];// 不能实例化一个泛型类型的数组// public T[] array = new T[10];//public T[] array = (T[])new Object[10];public Object[] array = new Object[10];public void setValue(int pos,T val) {array[pos] = val;}// 在java中不能将整个数组进行强转public T getValue(int pos) {return (T) array[pos];}}
class Person {}
public class Test {public static void main(String[] args) {MyArray<Integer> myArray = new MyArray<>();//MyArray<int> myArray = new MyArray<>(); //<>里面不能是基本数据类型myArray.setValue(1,90); // 直接放int 会发生自动装箱//裸类型  不加包装类型MyArray myArray1 = new MyArray<>();myArray1.setValue(0,1);}//目的:想存放指定的元素public static void main4(String[] args) {MyArray<String> myArray = new MyArray<>();myArray.setValue(0,"hello");MyArray<Integer> myArray2 = new MyArray<Integer>();//类型后加入<Integer> 指定当前类型myArray2.setValue(0,99);myArray2.setValue(1,20);MyArray<Person> myArray3 = new MyArray<>();}
}
  1. 类名后的 代表占位符,表示当前类是一个泛型类
    了解: 【规范】类型形参一般使用一个大写字母表示,常用的名称有:
  • E 表示 Element
  • K 表示 Key
  • V 表示 Value
  • N 表示 Number
  • T 表示 Type
  • S, U, V 等等 - 第二、第三、第四个类型
  1. 不能new泛型类型的数组
  2. 类型后加入 指定当前类型
  3. 不需要进行强制类型转换

4 泛型的使用

4.1 语法

//泛型类<类型实参> 变量名; // 定义一个泛型类引用
//new 泛型类<类型实参>(构造方法实参); // 实例化一个泛型类对象
MyArray<Integer> myArray = new MyArray<>();

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

4.2 类型推导

MyArray<Integer> list = new MyArray<>(); // 可以推导出实例化需要的类型实参为 Integer

5 裸类型

        //裸类型  不加包装类型MyArray myArray1 = new MyArray<>();myArray1.setValue(0,1);

总结

  1. 泛型是将数据类型参数化,进行传递
  2. 使用表示当前类是一个泛型类
  3. 泛型的优点:数据类型参数化,编译时自动进行类型检查和转换

6 泛型如何编译

6.1 擦除机制

在终端通过命令:javap -c 查看字节码文件,所有的T都是Object。
在这里插入图片描述

在编译的过程,将所有的T替换为Object这种机制,称为:擦除机制
即可以直接写成:

class MyArray <T>{//public T[] array = (T[])new Object[10];public Object[] array = new Object[10];public void setValue(int pos,T val) {array[pos] = val;}public T getValue(int pos) {return (T) array[pos]; //加(T) 转一下}public T[] getArray() {return (T[]) array;}}

7 泛型的上界

class MyArray <T extends Number>{  // T 一定是Number或者是Number的子类如Integerpublic Object[] array = new Object[10];public void setValue(int pos,T val) {array[pos] = val;}public T getValue(int pos) {return (T) array[pos];}
}public static void main(String[] args) {MyArray<String> myArray = new MyArray<>();  // 报错 String不是Number的子类// String[] ret = (String[])myArray.getArray();  //会报错 在java中不能将整个数组进行强转// 数组是一种单独的数据类型Object[] ret = myArray.getArray();}

写一个泛型类, 求一个数组当中的最大值

//写一个泛型类, 求一个数组当中的最大值
class Alg<T extends Comparable<T>> {  // 泛型的上界public T findMaxVal(T[] array) {T max = array[0];for (int i = 0; i < array.length; i++) {//if (array[i] > max) {  // 引用类型不能直接通过 大于等于号进行比较//这个时候就需要 写一个Comparable<T> 上界if (array[i].compareTo(max)>0){max = array[i];}}return max;}
}
public class Test {public static void main(String[] args) {Integer[] array = {1,2,8,5,3};Alg<Integer> alg = new Alg<>();System.out.println(alg.findMaxVal(array)); }
}

另一个类作为参数实例化时要引用它对应的接口和重写对应的方法

class Person implements Comparable<Person> {@Overridepublic int compareTo(Person o) {return 0;}
}//写一个泛型类, 求一个数组当中的最大值
class Alg<T extends Comparable<T>> {  // 泛型的上界public T findMaxVal(T[] array) {T max = array[0];for (int i = 0; i < array.length; i++) {//if (array[i] > max) {  // 引用类型不能直接通过 大于等于号进行比较//这个时候就需要 写一个Comparable<T> 上界if (array[i].compareTo(max)>0){max = array[i];}}return max;}
}
public class Test {public static void main(String[] args) {Integer[] array = {1,2,8,5,3};Alg<Integer> alg = new Alg<>();System.out.println(alg.findMaxVal(array));//如果定义一个类 作为Alg实例的话 这个类必须实现Comparable接口并且重写compareTo方法Alg<Person> alg1 = new Alg<Person>();}
}

8 泛型方法

在定义方法的时候加

class Alg2 {public<T extends Comparable> T findMaxVal(T[] array) {T max = array[0];for (int i = 0; i < array.length; i++) {if (array[i].compareTo(max)>0){max = array[i];}}return max;}
}public class Test {public static void main(String[] args) {Alg2 alg2 = new Alg2();Integer[] array = {1,2,8,5,3};alg2.<Integer>findMaxVal(array);  //可以不加<Integer>会通过实参的值自动推导alg2.findMaxVal(array);}
}

设为静态方法static

class Alg3 {public static <T extends Comparable> T findMaxVal(T[] array) {T max = array[0];for (int i = 0; i < array.length; i++) {if (array[i].compareTo(max)>0){max = array[i];}}return max;}
}public class Test {public static void main(String[] args) {Integer[] array = {1,2,8,5,3};Alg3.<Integer>findMaxVal(array);  //直接通过类名调用方法}
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

IP代理的巨大潜力,为什么跨境业务需要它?

IP说简单不简单&#xff0c;说复杂也不复杂&#xff0c;打个比方&#xff0c;IP就好比我们上网的一个门牌号&#xff0c;每家每户都会有一个门牌号&#xff0c;而且是唯一的地址。而代理IP&#xff08;代理服务器&#xff09;是一个位于中间的服务器&#xff0c;充当客户端和目…

【活动回顾】ABeam 德硕| 艾宾信息技术开发(西安)西北高校行——与西北三所高校签订校企合作协议

前言 INTRODUCTION 10月下旬&#xff0c;ABeam旗下艾宾信息技术开发&#xff08;西安&#xff09;校招团队来到宁夏大学、青海大学、兰州大学这三所高校&#xff0c;就校企合作达成多项共识并举行了隆重的签约仪式。ABeam大中华区董事长兼总经理中野洋辅先生也特意留出时间莅临…

使用conan包 - 安装依赖项

使用conan包 - 安装依赖项 主目录 conan Using packages1 Requires2 Optional user/channel3 Overriding requirements4 Generators5 Options 本文是基于对conan官方文档Installing dependencies的翻译而来&#xff0c; 更详细的信息可以去查阅conan官方文档。 This section s…

【shell脚本】常见的shell脚本面试题目

1、请用shell脚本for,while,until这三种方式写出输出1到100的所有偶数的方法。 sum0;for((i0;i<100;i2));do let sumi;done;echo $sum sum0;i0;while [ $i -le 100 ];do let sumi;let i2;done;echo $sum sum0;i0;until [ $i -gt 100 ];do let sumi;let i2;done;echo $sum#!…

【leetcode每日一题】565数组嵌套

思路流程&#xff1a; 思路v1.0 先学会写 s[0] ,用一个ans数组接收元素&#xff0c;每次往ans里添加的时候&#xff0c;先判断一下 这个index会不会超出数组的长度。ans里有没有这个元素。 s[0] 写完&#xff0c;就是用一个for循环&#xff0c;算出所有的 s[i],每次算出来的时…

野火霸天虎 STM32F407 学习笔记(六)系统时钟详解

STM32 中级 前言 仍然是学习自野火F407网课。 启动文件详解 作用&#xff1a; 初始化堆栈指针 SP_initial_sp初始化 PC 指针 Reset_Handler初始化中断向量表配置系统时钟调用 C 库函数 _main 初始化用户堆栈&#xff0c;从而最终调用 main 函数去到 C 的世界 栈&#xff…

School training competition ( Second )

A. Medium Number 链接 : Problem - 1760A - Codeforces 就是求三个数的中位数 : #include<bits/stdc.h> #define IOS ios::sync_with_stdio(0);cin.tie(0);cout.tie(0); #define endl \nusing namespace std; typedef long long LL; const int N 2e510;inline void …

Java线程同步

认识线程同步 解决方案 方法一&#xff1a;同步代码块 package com.itheima.d3;public class ThreadTest {public static void main(String[] args) {Accout acc new Accout("ICBC-110",100000);new DrawThread(acc,"小明").start();//小明new DrawThread…

鼎盛合:如何做一个车载便携吸尘器方案

汽车是我们日常生活中极为重要的交通工具&#xff0c;随着我们生活水平的提高自然而然地对汽车的要求也在相对提高&#xff0c;尤其是对汽车内部清洁更是引起许多有车人士的注意。如车载吸尘器深受大家的喜爱这一点就可见对汽车内部清洁的重视。 车载吸尘器是一个用于汽车清洁的…

Qt::Window 、Qt::Tool是 Qt 框架中的一个窗口标志(Window Flag),用于指定窗口的类型和行为

Qt::Window Qt::Window 是 Qt 框架中的一个窗口标志&#xff08;Window Flag&#xff09;&#xff0c;用于指定窗口的类型和行为。 在 Qt 中&#xff0c;窗口标志用于控制窗口的外观、行为和交互方式。通过使用不同的窗口标志组合&#xff0c;可以定制窗口的特性&#xff0c;…

Python实现DDos攻击实例详解

文章目录 SYN 泛洪攻击Scapy3k 基本用法代码实现DDos 实现思路argparse 模块socket 模块代码实现Client 端程序测试后记关于Python技术储备一、Python所有方向的学习路线二、Python基础学习视频三、精品Python学习书籍四、Python工具包项目源码合集①Python工具包②Python实战案…

Kotlin学习——kt里的集合,Map的各种方法之String篇

Kotlin 是一门现代但已成熟的编程语言&#xff0c;旨在让开发人员更幸福快乐。 它简洁、安全、可与 Java 及其他语言互操作&#xff0c;并提供了多种方式在多个平台间复用代码&#xff0c;以实现高效编程。 https://play.kotlinlang.org/byExample/01_introduction/02_Functio…

C++——vector插入与删除和数据存取

一. vector插入和删除 功能描述:对vector容器进行插入、删除操作 函数原型: push back(ele); //尾部插入元素ele pop_back(); //删除最后一个元素 insert(const iterator pos, el…

MFC 读写注册表

在MFC (Microsoft Foundation Classes) 中读写注册表涉及到使用 CRegKey 类&#xff0c;这个类提供了一组方法来操作Windows注册表。以下是如何使用MFC来读取和写入注册表值的基本步骤&#xff1a; 写入注册表值 创建或打开注册表项: 使用 CRegKey::Create 或 CRegKey::Open 方…

LuatOS-SOC接口文档(air780E)--rsa - RSA加密解密

示例 -- 请在电脑上生成私钥和公钥, 当前最高支持4096bit, 一般来说2048bit就够用了 -- openssl genrsa -out privkey.pem 2048 -- openssl rsa -in privkey.pem -pubout -out public.pem -- privkey.pem 是私钥, public.pem 是公钥 -- 私钥用于 加密 和 签名, 通常保密, 放在…

acwing算法基础之动态规划--背包问题

目录 1 基础知识2 模板3 工程化 1 基础知识 &#xff08;零&#xff09; 背包问题描述&#xff1a;有 N N N个物品&#xff0c;每个物品的体积是 v i v_i vi​&#xff0c;价值是 w i w_i wi​&#xff0c;现有容量是 V V V的背包&#xff0c;求这个背包能装下的物品的最大价值…

C#中的警告CS0120、CS0176、CS0183、CS0618、CS0649、CS8600、CS8601、CS8602、CS8604、CS8625及处理

目录 一、CS0120 二、CS0176 1.解决前 2.解决后 3.解决办法 三、CS0183 四、CS0618 五、CS8600 六、CS8602 七、CS8622 1. 解决前&#xff1a; 2. 解决后&#xff1a; 3.解决方法&#xff1a; 八、CS8604和CS8625 九、CS0649 十、CS8601 一、CS0120 严重性 代…

【算法萌新闯力扣】:回文链表

力扣题目&#xff1a;回文链表 开篇 今天是备战蓝桥杯的第23天。我加入的编程导航算法通关村也在今天开营啦&#xff01;那从现在起&#xff0c;我的算法题更新会按照算法村的给的路线更新&#xff0c;更加系统。大家也可以关注我新开的专栏“算法通关村”。里面会有更全面的知…

pandas教程:Techniques for Method Chaining 方法链接的技巧

文章目录 12.3 Techniques for Method Chaining&#xff08;方法链接的技巧&#xff09;1 The pipe Method&#xff08;pipe方法&#xff09; 12.3 Techniques for Method Chaining&#xff08;方法链接的技巧&#xff09; 对序列进行转换的时候&#xff0c;我们会发现会创建很…

操作系统的中断与异常(408常考点)

为了进行核心态和用户态两种状态的切换&#xff0c;引入了中断机制。 中断是计算机系统中的一种事件&#xff0c;它会打断CPU当前正在执行的程序&#xff0c;转而执行另一个程序或者执行特定的处理程序。中断可以来自外部设备&#xff08;如键盘、鼠标、网络等&#xff09;、软…