Java集合框架(包装类、泛型)

前言:

        本篇文章我们来讲解Java中的集合框架,就相当于车轮子。Java是面向对象的语言,所以相对于C语言有自身优势,就比如现成的数据结构(比如栈,队列,堆等)。Java的集合框架大家也不用想的很难,其实也就是这些内容。

        在了解集合框架之前,还需要一些预备知识,比如泛型(当然本篇文章是基础泛型讲解,对于初学者绝对够用),包装类等。

集合框架:

        Java集合框架Java Collection Framework,又称容器container,是定义在java.util包下的一组几口interfaces和其实现类classes。

        这张图说明了Java中类与类,类与接口之间的关系。这只是部分重要常见的类。

        我们可以看出,都是通过接口和类来使用的, 重要的有4个接口:List、Queue、Set、Map。其他类都是实现了这个接口。

        这里我们都来粗略的了解一下都是些什么:Stack是栈,ArrayList底层是动态链表(顺序表),LinkedList底层是双向链表(队列),PriorityQueue底层是优先队列,TreeSet、TreeMap底层是红黑树,HashSet、HashMap底层是哈希表(数组+链表+红黑树)。

        Set是集合,是一个接口。

包装类:

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

        几乎所有类型对应的包装类都是首字母大写就是其包装类,有两个例外:int对应包装类为Integer;char对应包装类为Character。

拆箱和装箱: 

        装箱和拆箱也叫装包和拆包。

public static void main(String[] args) {Integer a = 10;//装包int i = 99;Integer b = i;//也叫装包//基本类型转换为 包装类型System.out.println(a);System.out.println(b);
}

        我们执行完以后打开out目录并输入cmd,使用编辑模式观察。

        此时我们就进入源码来观察。 点击Integer,并搜索valueOf。

        装箱也分为自动装箱和显示装箱。 

public static void main(String[] args) {Integer a = 10;//装包  自动装箱int i = 99;Integer b = i;//也叫装包//基本类型转换为 包装类型System.out.println(a);System.out.println(b);Integer aa = Integer.valueOf(10);//显示装箱
}

        我们再来举个例子: 

public static void main(String[] args) {Integer a= 10;//装箱int i = a;    //拆箱System.out.println(i);int aa = a.intValue();//显示拆箱
}

        此时我们再观察以下代码: 

public static void main(String[] args) {Integer a = 100;Integer b = 100;System.out.println(a == b);Integer a1 = 200;Integer b1 = 200;System.out.println(a1 == b1);
}

        两个都是包装类型,那么你肯定感觉结果都是为:true。

        但是并不是,我们执行发现结果如下:

        这个结果很奇怪,我们只能进入源码观察。发现是low <= i <= high,所以我们要观察这两个值。

        

        所以high是127,low是-128。所以当我们的 i 在{-128,127}是不会新建对象的,而不在这个范围就会新建,而且直接比较对象是比较地址的,新建的话就会产生新的地址,从而不相等。

泛型: 

        泛型是一种语法。学过C语言的应该都知道(希望都学过,原谅作者但不影响阅读),我们可以定义数组,但是数组一定要明确里面存放的是什么类型的数据。比如我们创建一个栈(其实也就是数组),但是我们每次都要实现一个栈,必须明确这个栈里面存放的是什么类型的数据,这就很麻烦,于是Java中就引入了泛型的定义。

        这里,请允许我举一个例子:比如此时我们引出一个类,类中包含一个数组成员,是的数组中可以存放任何类型的数据,也可以根据成员方法返回数组中某个下标的值。

class MyArray {public Object[] array = new Object[10];public void setValue(int pos, Object val) {array[pos] = val;}public Object getValue(int pos) {return array[pos];}
}public class Test2 {public static void main(String[] args) {MyArray myArray = new MyArray();//此时就可以存放任何数据类型了myArray.setValue(0,10);myArray.setValue(1,"hello");String str = (String)myArray.getValue(1);//此时我们知道下标 1 放的是 String 类型}
}

        你可以把这个理解为泛型的雏形,因为你确实可以这样写,但是我们一眼就可以发现问题:这个弊端很大,我们要知道每一个下标放的是什么,但是自己写的自己知道,给别人就会迷糊。

        所以我们引入了泛型,使用<>来指定存放的什么类型。比如:

         此时我们还是要借助Object类:

//<T> 当前类 是一个泛型类 它只是一个占位符
class MyArray<T> {//public T[] array = new T[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];}
}public class Test2 {public static void main(String[] args) {MyArray<Integer> myArray = new MyArray<Integer>();myArray.setValue(0,10);myArray.setValue(1,20);int a = myArray.getValue(1);//不用进行强制类型转换System.out.println(a);MyArray<String> myArray1 = new MyArray<String >();myArray1.setValue(0,"abcd");myArray1.setValue(1,"efg");String ret = myArray1.getValue(1);System.out.println(ret);}
}

        我们就知道该使用Object时还是要使用,但是泛型还是方便了我们的使用,防止我们出错。此时我们也就明白了包装类的意义,因为我们不能直接传入整形。

MyArray<Integer> myArray = new MyArray<>();//此时就可以省略后面的泛型
myArray.setValue(0,10);
myArray.setValue(1,20);
int a = myArray.getValue(1);//不用进行强制类型转换
System.out.println(a);

        我们不能new泛型的对象(new T)。在编译时期泛型是存在的,当程序运行起来到JVM后,就没有泛型的概念了。

        泛型在编译的时候如何编译?使用过擦除机制,擦除成了Object。Java中不允许直接返回泛型数组。

泛型注意事项: 

class MyArray <T> {public T[] ts1 = new T[2];//这样写不被允许public T[] ts2 = (T[]) new Object[2];
}

 

        当返回的是一个泛型的数组时,还是需要借助Object类来完成。 

        尖括号里面必须放引用类型。

<T extends Number>:

//T 一定是 Number 的子类
class TestGeneric<T extends Number> {}public class Test {public static void main(String[] args) {TestGeneric<Number> testGeneric1 = new TestGeneric<>();TestGeneric<Integer> testGeneric2 = new TestGeneric<>();TestGeneric<Double> testGeneric3 = new TestGeneric<>();TestGeneric<String> testGeneric4 = new TestGeneric<>();}
}

        泛型的上界, 泛型是没有下界的。

练习题:

        写一个泛型类,求一个数组中的最大值。T一定是引用类型,最终被擦除为了Object类型。而Object类型没有实现Comparable接口。

class Alg<T> {public T findMaxValue(T[] array) {T max = array[0];for (int i = 1; i < array.length; i++) {if (max < array[i]) {max = array[i];}}return max;}
}

        报错是因为我们没有指定一个类型作比较。 T类型一定是可以比较的,问题是怎么能够约束这个T一定是可以比较大小的?

class Alg<T extends Comparable<T>>

        这句不是说T继承Comparable,而是说将来指定传入的类一定实现了Comparable接口。

class Alg<T extends Comparable<T>> {public T findMaxValue(T[] array) {T max = array[0];for (int i = 1; i < array.length; i++) {if (max.compareTo(array[i]) < 0) {max = array[i];}}return max;}
}public class Test {public static void main(String[] args) {Alg<Integer> alg = new Alg<>();Integer[] integers = {1, 2, 3, 4, 5, 6};Integer ret = alg.findMaxValue(integers);System.out.println(ret);}
}

        包装类Integer实现了这个接口,所以我们可以传入整形数组。 传入数组是因为使用泛型本身就是要使用引用类型。

泛型方法:

class Alg2 {public <T extends Comparable<T>> T findMaxValue(T[] array) {T max = array[0];for (int i = 1; i < array.length; i++) {if (max.compareTo(array[i]) < 0) {max = array[i];}}return max;}
}public class Test {public static void main(String[] args) {Alg2 alg2 = new Alg2();Integer[] integers = {1, 2, 3, 4, 5, 6};Integer ret = alg2.findMaxValue(integers);System.out.println(ret);//6}
}

         此时发生了类型推导,根据实参传值,来推导此时的类型。也可以写上去:

Integer ret = alg2.<Integer>findMaxValue(integers);

总结:

        本篇作者其实有些敷衍,并没有考虑到小白,可以说这是一篇笔记,但是相信对于一些人还是有帮助的,也希望各位体谅。

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

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

相关文章

13 年后,我如何用 Go 编写 HTTP 服务(译)

原文&#xff1a;Mat Ryer - 2024.02.09 大约六年前&#xff0c;我写了一篇博客文章&#xff0c;概述了我是如何用 Go 编写 HTTP 服务的&#xff0c;现在我再次告诉你&#xff0c;我是如何写 HTTP 服务的。 那篇原始的文章引发了一些热烈的讨论&#xff0c;这些讨论影响了我今…

2019 年全国职业院校技能大赛高职组 “信息安全管理与评估”赛项任务书(笔记详解)

1. 网络拓扑图 2. IP 地址规划表 3. 设备初始化信息 阶段一 任务 1:网络平台搭建 1、根据网络拓扑图所示,按照 IP 地址参数表,对 DCFW 的名称、各接口IP 地址进行配置。 2、根据网络拓扑图所示,按照 IP 地址参数表,对 DCRS 的名称进行配置,创建 VLAN 并将相应接口划入 …

二十、K8S-1-权限管理RBAC详解

目录 k8s RBAC 权限管理详解 一、简介 二、用户分类 1、普通用户 2、ServiceAccount 三、k8s角色&角色绑定 1、授权介绍&#xff1a; 1.1 定义角色&#xff1a; 1.2 绑定角色&#xff1a; 1.3主体&#xff08;subject&#xff09; 2、角色&#xff08;Role和Cluster…

第三百一十六回

[tod] 我们在上一章回中介绍了"如何在输入框中处理光标"相关的内容&#xff0c;本章回中将介绍如何添加输入框默认值.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概念介绍 在项目中经常使用输入框获取用户输入的内容&#xff0c;有时候在输入框中反复输入相…

城市合伙人计划:资源共享、知识交流、合作创新

城市合伙人计划是一种合作伙伴关系&#xff0c;旨在通过共享资源、知识和经验&#xff0c;共同推动城市的经济发展和社会进步。这种计划通常涉及市政府、企业、社会组织和个人等多个方面&#xff0c;通过合作共同解决城市面临的问题和挑战。 城市合伙人计划的具体形式和内容可…

基于SSM的图书管理系统

点击以下链接获取源码&#xff1a; https://download.csdn.net/download/qq_64505944/88825395?spm1001.2014.3001.5503 Java项目-14 1、导入源码 不成功导入模块也可以 2、配置jdk-一般为1.8 3、修改文件中数据库连接名与密码 4、配置Maven 5、更新Maven 7、添加数据库…

【MySQL】-12 MySQL索引(上篇MySQL索引类型前置-1)

MySQL索引 索引1 索引基础2 索引与优化1 选择索引的数据类型1.1 选择标识符 2 索引入门2.1 索引的类型2.1.1 B-Tree索引2.1.2 Hash索引2.1.3 空间(R-Tree)索引2.1.4 全文(Full-text)索引 索引的优点&#xff1a;索引是最好的解决方案吗&#xff1f; 索引 索引&#xff08;在MYS…

【笔记】Harmony学习:下载安装 DevEco Studio 开发工具IDE

IDE 安装 从官网下载DevEco Studio 安装包后进行安装&#xff0c; 安装完毕后&#xff0c;本地环境可能要配置相关工具&#xff0c;可以通过下面的诊断检测一下本地环境&#xff0c;通过蓝色“Set it up now” 可以快速安装。 1. Node.js (for ohpm) 2. ohpm 下载op的包管理&a…

精灵图,字体图标,CSS3三角

精灵图 1.1为什么需要精灵图 一个网页中往往会应用很多小的背景图像作为修饰&#xff0c;当网页中的图像过多时&#xff0c;服务器就会频繁的接受和发送请求图片&#xff0c;造成服务器请求压力过大&#xff0c;这将大大降低页面的加载速度。 因此&#xff0c;为了有效地减少…

43.1k star, 免费开源的 markdown 编辑器

简介 项目名&#xff1a; MarkText-- 简单而优雅的开源 Markdown 编辑器 Github 开源地址&#xff1a; https://github.com/marktext/marktext 官网&#xff1a; https://www.marktext.cc/ 支持平台&#xff1a; Linux, macOS 以及 Windows。 操作界面&#xff1a; 在操作界…

一场由对生成型人工智能的普遍不满引发的全面攻击正在展开

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

【Linux】学习-动静态库

动静态库 头文件与库的区别 头文件一般而言&#xff0c;是声明和宏定义。头文件是在预处理阶段使用的 库文件是已经编译好的二进制代码。是一种目标文件&#xff0c;库文件是在链接阶段使用的 对于头文件和库我们可以这样理解&#xff0c;就是头文件提供的是一个函数的声明&…

使用Flash download tool进行ESP32固件烧录

背景 为方便分发固件&#xff0c;可在任意电脑上安装烧录软件&#xff0c;直接将固件烧录进 烧录内容 查看vscode上platformio的烧录过程 Writing at 0x00000000... (100 %) Wrote 15104 bytes (10401 compressed) at 0x00000000 in 0.4 seconds (effective 281.3 kbit/s).…

Pandas常用操作记录(更新中)

1.读取文件 import pandas as pd df pd.read_csv(路径) #pd.read_文件格式(路径) 2.读取某列某行&#xff0c;并使用map替换 2.1 直接读取某列数据 在获取到df对象后&#xff0c;可以使用 df.列名 来获取该列数据&#xff0c; import pandas as pd df pd.read_csv(rdat…

电商小程序06用户审核

目录 1 创建自定义应用2 显示待办数量3 创建审核页面4 开发审核功能5 搭建布局6 最终效果总结 上一篇我们讲解了用户注册的功能&#xff0c;用户注册之后状态是待审核&#xff0c;需要管理员进行审核。通常给管理员提供一套PC端的软件进行相关的操作&#xff0c;在低代码中&…

ChatGPT高效提问—prompt常见用法(续篇五)

ChatGPT高效提问—prompt常见用法&#xff08;续篇五&#xff09; 1.1 种子词 ​ 种子词&#xff08;seed word&#xff09;通常指的是在对话中使用的初始提示或关键词&#xff0c;用于引导ChatGPT生成相关回复。种子词可以是一个词、短语或句子&#xff0c;通常与对话的主题…

代码随想录算法训练营第四十八天(动态规划篇之01背包)| 1049. 最后一块石头的重量Ⅱ,494. 目标和

1049. 最后一块石头的重量Ⅱ 题目链接&#xff1a;1049. 最后一块石头的重量 II - 力扣&#xff08;LeetCode&#xff09; 思路 尽量将石头分为重量相同的两堆&#xff0c;这样两堆中的石头相撞之后剩下的石头就会最小。根据之前的01背包理论&#xff1a; 代码随想录算法训…

【Chrono Engine学习总结】3-地型terrain

由于Chrono的官方教程在一些细节方面解释的并不清楚&#xff0c;自己做了一些尝试&#xff0c;做学习总结。 1、关于物体材质 在介绍地型之前&#xff0c;要初步了解chrono中关于材质的一些基本概念。 首先&#xff0c;最基本的材质类是ChMaterialSurface,其进一步包括&…

原生JS使用PrintJs进行表格打印 -- 遇到的问题总结

需求1&#xff1a;表格自动分页之后&#xff0c;表头在每一页都需要显示 html中表头增加 thead 标签 css样式新增&#xff1a; thead {display: table-header-group; /* 这个属性使thead总是在新的page-break之后重新开始 */ }需求2&#xff1a;表格自动分页之后&#xff0c;…

数据可视化之维恩图 Venn diagram

文章目录 一、前言二、主要内容三、总结 &#x1f349; CSDN 叶庭云&#xff1a;https://yetingyun.blog.csdn.net/ 一、前言 维恩图&#xff08;Venn diagram&#xff09;&#xff0c;也叫文氏图或韦恩图&#xff0c;是一种关系型图表&#xff0c;用于显示元素集合之间的重叠区…