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://xiahunao.cn/news/2777397.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

使用AI开发一个红包封面生成器

使用 VUE3&#xff0c;和 Express 开发一个红包封面。 生成效果如下 体验地址&#xff1a;https://hongbao.digitalmodel.top/

Web Services 服务 是不是过时了?创建 Web Services 服务实例

Web Services 是不是过时了&#xff1f; 今天是兔年最后一天&#xff0c;先给大家拜个早年 。 昨天上午视频面试一家公司需要开发Web Services 服务&#xff0c;这个也没有什么&#xff0c;但还需要用 VB.net 开发。这个是多古老的语言了&#xff0c;让我想起来了 10年 前 写 …

无人机应用场景和发展趋势,无人机技术的未来发展趋势分析

随着科技的不断发展&#xff0c;无人机技术也逐渐走进了人们的生活和工作中。无人机被广泛应用于很多领域&#xff0c;例如遥感、民用、军事等等。本文将围绕无人机技术的应用场景和发展趋势&#xff0c;从多角度展开分析。 无人机技术的应用场景 无人机在遥感方面的应用&…

C++之RTTI实现原理

相关系列文章 C无锁队列的原理与实现 如何写出高质量的函数&#xff1f;快来学习这些coding技巧 从C容器中获取存储数据的类型 C之多层 if-else-if 结构优化(一) C之多层 if-else-if 结构优化(二) C之多层 if-else-if 结构优化(三) C之Pimpl惯用法 C之RTTI实现原理 目录 1.引言…

Swift Combine 使用 dataTaskPublisher 发起网络请求 从入门到精通十

Combine 系列 Swift Combine 从入门到精通一Swift Combine 发布者订阅者操作者 从入门到精通二Swift Combine 管道 从入门到精通三Swift Combine 发布者publisher的生命周期 从入门到精通四Swift Combine 操作符operations和Subjects发布者的生命周期 从入门到精通五Swift Com…

旅游|基于Springboot的旅游管理系统设计与实现(源码+数据库+文档)

旅游管理系统目录 目录 基于Springboot的旅游管理系统设计与实现 一、前言 二、系统功能设计 三、系统实现 1、用户管理 2、景点分类管理 3、景点信息管理 4、酒店信息管理 5、景点信息 6、游记分享管理 四、数据库设计 1、实体ER图 2、具体的表设计如下所示&#xf…

【从Python基础到深度学习】4. Linux 常用命令

1.配置root用户密码 root用户为系统默认最高权限用户&#xff0c;其他用户密码修改命令与root用户修改密码命令相同 sudo passwd root 2.添加用户&#xff08;henry&#xff09; sudo useradd -m henry -s /bin/bash 3.配置henry用户密码 Xshell下连接新用户&#xff08;hen…

小巨人大爆发:紧凑型大型语言模型效率之谜揭晓!

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

图像处理常用算法—6个算子 !!

目录 前言 1、Sobel 算子 2、Isotropic Sobel 算子 3、Roberts 算子 4、Prewitt 算子 5、Laplacian算子 6、Canny算子 前言 同图像灰度不同&#xff0c;边界处一般会有明显的边缘&#xff0c;利用此特征可以分割图像。 需要说明的是&#xff1a;边缘和物体间的边界并不…

Django问题报错:TypeError: as_view() takes 1 positional argument but 2 were given

一、错误位置 from django.urls import pathfrom users_app.views import RegisterView, LoginView, LogoutViewapp_name users urlpatterns [path("register/", RegisterView.as_view, name"register"),path("login/", LoginView.as_view, n…

机器学习---学习与推断,近似推断、话题模型

1. 学习与推断 基于概率图模型定义的分布&#xff0c;能对目标变量的边际分布&#xff08;marginal distribution&#xff09;或某些可观测变量 为条件的条件分布进行推断。对概率图模型&#xff0c;还需确定具体分布的参数&#xff0c;称为参数估计或学习问 题&#xff0c;…

读千脑智能笔记08_人工智能的未来(下)

1. 机器智能存在的风险 1.1. “人工智能”这个名字应用到几乎所有涉及机器学习的领域 1.2. 技术专家对人工智能的态度也从“人工智能可能永远不会实现”快速转变为“人工智能可能在不久的将来毁灭所有人类” 1.3. 每一项新技术都可能会被滥用…

专业课135+总分400+西安交通大学815/909信号与系统考研电子信息与通信工程,真题,大纲,参考书。

经过将近一年的考研复习&#xff0c;终于梦圆西安交大&#xff0c;今年专业可815(和909差不多)信号与系统135&#xff0c;总分400&#xff0c;回想这一年的复习还是有很多经验和大家分享&#xff0c;希望可以对大家复习有所帮助&#xff0c;少走弯路。 专业课&#xff1a; 这…

18:蜂鸣器

蜂鸣器 1、蜂鸣器的介绍2、编程让蜂鸣器响起来3、通过定时控制蜂鸣器4、蜂鸣器发出滴滴声&#xff08;间歇性鸣叫&#xff09; 1、蜂鸣器的介绍 蜂鸣器内部其实是2个金属片&#xff0c;当一个金属片接正电&#xff0c;一个金属片接负电时&#xff0c;2个金属片将合拢&#xff…

大数据应用对企业的价值

目录 一、大数据应用价值 1.1 大数据技术分析 1.2 原有技术场景的优化 1.2.1 数据分析优化 1.2.2 高并发数据处理 1.3 通过大数据构建新需求 1.3.1 智能推荐 1.3.2 广告系统 1.3.3 产品/流程优化 1.3.4 异常检测 1.3.5 智能管理 1.3.6 人工智能和机器学习 二、大数…

【深度学习: ChatGPT 】经验教训:使用 ChatGPT 作为 ML 工程师一天

【深度学习&#xff1a; ChatGPT 】经验教训&#xff1a;使用 ChatGPT 作为 ML 工程师一天 介绍设置过程标杆ChatGPT 做机器学习ChatGPT 能否真正实施这些解决方案&#xff1f;结果结论 TLDR;在最近使用 AI 应用程序 ChatGPT 的用例激增中&#xff0c;我们询问它是否可用于改进…

肯尼斯·里科《C和指针》第12章 使用结构和指针(1)链表

只恨当时学的时候没有读到这本书&#xff0c;&#xff0c;&#xff0c;&#xff0c;&#xff0c;&#xff0c; 12.1 链表 有些读者可能还不熟悉链表&#xff0c;这里对它作一简单介绍。链表(linked list)就一些包含数据的独立数据结构&#xff08;通常称为节点&#xff09;的集…

【数学建模】【2024年】【第40届】【MCM/ICM】【A题 七鳃鳗性别比与资源可用性】【解题思路】

我们通过将近半天的搜索数据&#xff0c;查到了美国五大湖中优势物种的食物网数据&#xff0c;以Eric伊利湖为例&#xff0c;共包含34各优势物种&#xff0c;相互之间的关系如下图所示&#xff1a; 一、题目 &#xff08;一&#xff09; 赛题原文 2024 MCM Problem A: Reso…

704. Binary Search(二分查找)

题目描述 给定一个 n 个元素有序的&#xff08;升序&#xff09;整型数组 nums 和一个目标值 target &#xff0c;写一个函数搜索 nums 中的 target&#xff0c;如果目标值存在返回下标&#xff0c;否则返回 -1。 问题分析 确定左右界&#xff0c;然后按规则进行更新即可 代…

H12-821_73

73.某台路由器Router LSA如图所示&#xff0c;下列说法中错误的是&#xff1f; A.本路由器的Router ID为10.0.12.1 B.本路由器为DR C.本路由器已建立邻接关系 D.本路由器支持外部路由引入 答案&#xff1a;B 注释&#xff1a; LSA中的链路信息Link ID&#xff0c;Data&#xf…