Java集合-List(Collection子接口)及其子类(ArrayList、Vector、LinkedList)

List接口是 Collection接口的子接口。
1、List集合类中数据有序, 即添加顺序和取出顺序有序,而且可以重复。
2、List集合类中每个元素都有其对应的顺序索引,即支持索引。例,list.get(2);取第三个元素。
3、实现类有很多,介绍ArrayList、Vector、LinkedList三种。
常用方法:
其方法相当于在父接口Collection中加入了索引,可以根据索引进行增删改查。
1、void add(int index, Object ele)
将指定的元素插入此列表中的指定位置(可选操作)。
该位置之后的元素则依次往后移一位。
2、boolean addAll(int index, Collection eles)
从index位置开始,将eles中的所有元素添加进来。
3、Object get(int index)
获取index位置的元素。
4、int indexOf(Object obj)
返回obj元素在集合中首次出现的位置。
5、int lastIndexOf(Object obj)
返回obj元素在集合中最后出现的位置。
6、Object remove(int index)
移除指定index位置的元素,并返回该元素。
7、Object set(int index, Object ele)
设置指定index位置的元素为ele,相当于用ele 替换原来元素。注意,index位置必须有元素,否则抛出异常。
8、List subList(int formIndex, int toIndex)
返回从 formIndex 到 toIndex 的子集合。
遍历方式:
因为其是Collection的子接口,所以方法1和方法2与Collection一致。
但是其可以有索引,因此通过索引访问,因此可以通过方法3,普通for循环通过索引访问。
1、使用迭代器Iterator
2、使用增强for
3、使用普通for

一、ArrayList

1、ArrayList可以添加null,并且可以添加多个。
2、ArrayList是由数组来实现数据存储的。
3、ArrayList基本等同于Vector,除了ArrayList是线程不安全的,多线程不建议使用ArrayList。

底层源码:

1、ArrayList 中维护了一个 Object 类型的数组 elementData[].
transient关键字:即被其修饰的属性不会被序列化。
2、创建ArrayList对象逻辑分析。
如果使用无参构造器,则elementData[]的容量为0;
第一次添加则扩容为10,如果需要再次扩容,则扩容为原来的1.5倍。
如果使用指定大小的构造器,则elementData[]的容量为指定大小,
如果需要扩容,则直接扩容为1.5倍。
①使用无参构造器创建对象:
给elementData赋初值 DEFAULTCAPACITY_EMPTY_ELEMENTDATA ,即第二张图:空数组。
指定大小创建对象:(需要看完无参构造创建对象的四个步骤再看)
即使用有参构造器ArrayList(int initialCapacity),
如果给定的值大于0,则直接给elementData赋值一个对应大小的数组
如果等于0,则跟使用无参一样。
否则抛出异常。
所以,只要使用无参构造器,所有的ArrayList的elementData属性都是同一个:
②第一次执行add()
先使用ensureCapacityInternal()函数判断容量够不够
然后在进行添加元素。
③确定容量是否足够
调用ensureCapacityInternal()函数,
调用calculateCapacity()函数返回一个最小容量。
得到最小容量后调用ensureExplicitCapacity()判断是否需要扩容。
详情如下:
ensureCapacityInternal()函数:
            
calculateCapacity()函数:
先确定elementData[]是否是最开始的空数组DEFAULTCAPACITY_EMPTY_ELEMENTDATA,
如果是,就返回 DEFAULT_CAPACITY 和 minCapacity 中较大的那个,
其中DEFAULT_CAPACITY初始值为10.(定义如下图二)
否则直接返回minCapacity。
ensureExplicitCapacity():
第一步modCount++,是记录当前集合被操作次数。
第二步判断最小容量 - 当前容量是否大于0,
如果>0,说明实际容量比最小容量要小,则容量不够了,需要进行扩容。
则调用grow()方法去扩容。
④使用grow()去扩容
先获取newCapacity的值,即新数组的容量,
然后使用Arrays.copyOf()函数获得新数组。
详细过程:
oldCapacity接收未扩容前的容量;
newCapacity的值为oldCapacity的1.5倍,即 newCapacity = oldCapacity + (oldCapacity >> 1);
因为第一次old为0,所以new也为0,因此需要判断一下:
当new - min < 0时,即新的容量小于最小容量,则直接将min赋值给new
如果新的容量比最大容量大,则进行hugeCapacity()方法
最大容量MAX_ARRAY_SIZE定义在图二
最后使用 Arrays.copyOf()函数获得新数组。

二、Vector

1、vector也是使用数组来实现存储的。
2、vector是线程同步的,即线程安全。

1、定义

其在底层也维护了一个Object类型的数组,用来存储数据:
可见没有transient关键字,即可以序列化。
其线程安全:
其方法都有synchronized关键字修饰。

2、与ArrayList比较:

3、创建对象逻辑分析(源码)

①无参
第一步:调用自身的有参构造器,默认赋值为10;
②add()
add()方法和ArrayList类似,只是将modCount++放在了最开始,
然后执行ensureCapacityHelper()方法确定容量是否足够。
如果不够则执行grow()方法
够则直接添加数据。
详细:
add():
ensureCapacityHelper():
grow():
可见与ArrayList不同:int newCapacity = oldCapacity + ((capacityIncrement > 0) ? capacityIncrement : oldCapacity);
先判断属性capacityIncrement是否大于0,若不大于0,则new = old +old,即扩容为2倍。
属性capacityIncrement作用在有参构造中使用,其默认值为0;
③有参
如下如,Vector的三个构造器,源码如下:
可见其为相互调用,第三个则为最终构造器。
第一个参数为初始容量
第二个参数最终赋给capacityIncrement属性,作用为说明每次容量满时,增加多少,如上图grow()函数。
即不给此变量赋值则增长为2倍。
 

三、LinkedList

1、底层实现了双向链表和双端队列。
2、可以添加任意元素,包括null。
3、线程不安全,没有实现同步。

1、底层操作机制:

1、底层维护了一个双向链表
2、维护两个属性first和last,分别指向首节点和尾节点。
3、每个节点(Node对象),里面维护了prev、next和item三个属性,分别是前驱,后驱和值。
注意Node为LinkedList的内部类。
4、因此LinkedList元素的删除添加不是通过数组完成,效率更高。
即不用新建数组,然后复制原数组到救数组。

2、与ArrayList对比

3、源码分析:

1、new():创建对象
①无参:
可见无参构造器什么也不做。
即仅仅做一个初始化。
其中first = null;  last = null;  size = 0;
②有参:
有参则先创建一个无参的,然后向里面添加对象。
2、add():增加对象
此处介绍没有index参数的增加,即向链表最后添加。
首先执行add()方法
然后执行linkLast()方法,将元素添加到链表最后。
分析linkLast()方法:
①用 l 来存储 last节点
②新建一个newNode
该节点的prev = l,item = e, next = null,即新建节点前驱为插入前的last节点,值为传进来的值,后驱节点为null;
③将last节点指向newNode
④判断:
若l = null,即第一次插入(因为只有第一次插入之前,last和first都为null),将first也指向newNode
若i ≠ null,说明之前依旧插入过,该链表当中已经有元素存在,则将l.next指向newNode,即 l 节点也就是原末节点变成了现末节点的前驱。
⑤最后把size加一,把modcount加一。
3、remove()删除对象
此处介绍没有参数的删除,即删除第一个元素。
首先 调用removeFirst()函数,本函数用来判断是否为null,
然后执行unlinkFirst()函数删除第一个元素。
可见, removeFirst()函数首先判断first是否为null,为空则抛出异常;
不为空则调用unlinkFirst()方法,并将f传进去。
首先,将节点f 的item赋给element,用以最后返回
next赋给next,用来作为新的first。
将节点f 的item 和 next均置为null,此时gc会判断其为垃圾,将其回收。
然后,让first指向next,即指向原来的第二个元素;
判断:
如果next = null,即原来就只有一个元素,删除的不仅是第一个元素,也是Last所指向的元素,所以需要将last也指向null。
如果next ≠ null,说明被移除的结点不是最后一个结点,此时被移除的结点的后一个结点对象持有的prev需要指向null,这样就断开了对移除结点的引用。   
最后将size减一,将modCount加一。
有参数删除remove(int index):
有参数int的话,会先调用checkElementIndex(index)函数判断对应节点是否存在(通过isElementIndex()判断index是否在0和size之间)
然后调用unlink(node(index))函数去删除,先使用node(index)函数以此获取到第index位置的节点
然后使用unlink(Node<E> x)函数删除对应元素。
4、set()修改和get()查询
这两个函数都是调用node()函数得到对应的节点然后进行操作,与上述remove()一致。

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

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

相关文章

家政预约小程序10公众号集成

目录 1 使用测试号3 工作流配置4 配置关注事件脚本5 注册开放平台6 获取公众号access_token6 实现关注业务逻辑总结 我们本次实战项目构建的相当于一个预约平台&#xff0c;既有家政企业&#xff0c;也有家政服务人员还有用户。不同的人员需要收到不同的消息&#xff0c;比如用…

99.网络游戏逆向分析与漏洞攻防-ui界面的设计-角色信息显示的界面与功能

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 如果看不懂、不知道现在做的什么&#xff0c;那就跟着做完看效果&#xff0c;代码看不懂是正常的&#xff0c;只要会抄就行&#xff0c;抄着抄着就能懂了 内容…

机器人学导论P115求雅可比矩阵python实现

代码如下&#xff1a; import numpy as np import matplotlib.pyplot as plt import seaborn as sns import plotly.express as px import plotly.graph_objects as go from plotly.subplots import make_subplots from numpy import sin from numpy import cos plt.rcParams[…

【考试100】安全员B证《建设工程安全生产技术》单选题

​ 题库来源&#xff1a;考试100 【考试100】安全员B证《建设工程安全生产技术》单选题 1&#xff0e;在悬空部位作业时&#xff0c;操作人员应&#xff08; &#xff09; A.遵守操作规定 B.进行安全技术交底 C.戴好安全帽 D.系好安全带 【考试100答案】&#xff1a;D…

【R基础】如何开始学习R-从下载R及Rstudio开始

文章目录 概要下载R流程下载Rstudio流程下载完成-打开 概要 提示&#xff1a;如何开始学习R-从下载R及Rstudio开始&#xff0c;此处我只是想下载指定版本R4.3.3 下载R流程 链接: R官网 文件下载到本地 下载文件展示 按照向导指示安装 下载Rstudio流程 链接: Rstudio官网…

低代码与人工智能的深度融合:行业应用的广泛前景

引言 在当今快速变化的数字化时代&#xff0c;企业面临着越来越多的挑战和机遇。低代码平台和人工智能技术的兴起&#xff0c;为企业提供了新的解决方案&#xff0c;加速了应用开发和智能化转型的步伐。 低代码平台的基本概念及发展背景 低代码平台是一种软件开发方法&#x…

解决MYSQL5.7版本only_full_group_by报错解决方法

问题 出现this is incompatible with sql_modeonly_full_group_by这个语句就说明启动了only_full_group_by规则了 介绍only_full_group_by规则&#xff1a; 这种情况可能是5.7版本的规则比较严格&#xff0c;当启用“only_full_group_by”模式时&#xff0c;MySQL会对执行GROU…

SpringBoot中MyBatisPlus的使用

MyBatis Plus 是 MyBatis 的增强工具&#xff0c;提供了许多强大的功能&#xff0c;简化了 MyBatis 的使用。下面是在 Spring Boot 中使用 MyBatis Plus 的步骤&#xff1a; 添加依赖&#xff1a;在 Maven 或 Gradle 的配置文件中添加 MyBatis Plus 的依赖。 配置数据源&#…

Day10:平面转换、渐变色

目标&#xff1a;使用位移、缩放、旋转、渐变效果丰富网页元素的呈现方式。 一、平面转换 1、简介 作用&#xff1a;为元素添加动态效果&#xff0c;一般与过渡配合使用。 概念&#xff1a;改变盒子在平面内的形态&#xff08;位移、旋转、缩放、倾斜&#xff09;。 平面转换…

小米投屏怎么投?收好这3个投屏指南!(2024新)

近年来&#xff0c;小米凭借过硬的品质和合理的价格成为手机市场的一股强劲力量。随着其销量的上升&#xff0c;人们可以通过多种方式使用它来获得乐趣和便利。比如小米MIUI 11自带一个“光环”——Miracast&#xff0c;可以让用户在电脑上控制小米/红米/小米&#xff0c;获得更…

The book

Deep Learning for Coders with Fastai and PyTorch: AI Applications Without a PhD is the book that forms the basis for this course. We recommend reading the book as you complete the course. There’s a few ways to read the book – you can buy it as a paper bo…

CameraProvider启动流程

从Android 8.0之后&#xff0c;Android 引入Treble机制&#xff0c;主要是为了解决目前Android 版本之间升级麻烦的问题&#xff0c;将OEM适配的部分vendor与google 对android 大框架升级的部分system部分做了分离&#xff0c;一旦适配了一个版本的vendor信息之后&#xff0c;之…

【excel】设置可变下拉菜单(一级联动下拉菜单)

文章目录 【需求】制作动态下拉菜单&#xff0c;显示无重复的“班级”列表【思路】设置辅助列&#xff0c;使用UNIQUE()函数去重&#xff0c;并用FILTER()去掉结果中的“0”【步骤】step1 辅助列step2 设置下拉菜单 【总结】 【需求】制作动态下拉菜单&#xff0c;显示无重复的…

深度学习之AlexNet、VGG-19、VGG-16、LeNet-5、ResNet模型的训练

一&#xff0e;AlexNet 1.1.导入资源包 import cv2 import matplotlib.pyplot as plt import numpy as np import os import random注&#xff1a; cv2&#xff1a;这是 OpenCV 模块&#xff0c;用于处理图像和视频&#xff0c;包括摄像头捕捉、图像处理、特征检测等。 matpl…

Playwright 自动化操作

之前有见同事用过playwright进行浏览器模拟操作&#xff0c;但是没有仔细了解&#xff0c;今天去详细看了下&#xff0c;发现playwright着实比selenium牛逼多了 Playwright 相对于selenium优点 1、自动下载chromnium, 无需担心chrome升级对应版本问题&#xff1b; 2、支持录屏操…

干Java的有4年的工作经验;想转行做labview能行吗?

在开始前刚好我有一些资料&#xff0c;是我根据网友给的问题精心整理了一份「 Java的资料从专业入门到高级教程」&#xff0c; 点个关注在评论区回复“888”之后私信回复“888”&#xff0c;全部无偿共享给大家&#xff01;&#xff01;&#xff01;bVIEW和Java都是软件工具&a…

推荐几款优秀的文档加密软件 | 企业文件加密解决方案

在数字化时代&#xff0c;信息安全问题日益突出&#xff0c;文档加密软件成为了保护数据安全的重要手段。但是&#xff0c;市面上的文档加密软件种类繁多&#xff0c;功能各异&#xff0c;如何选择一款好用的文档加密软件成为了许多用户关注的焦点。本文将为大家提供一份实用的…

【第十三节】C++控制台版本坦克大战小游戏

目录 一、游戏简介 1.1 游戏概述 1.2 知识点应用 1.3 实现功能 1.4 开发环境 二、项目设计 2.1 类的设计 2.2 各类功能 三、程序运行截图 3.1 游戏主菜单 3.2 游戏进行中 3.3 双人作战 3.4 编辑地图 一、游戏简介 1.1 游戏概述 本项目是一款基于C语言开发的控制台…

5. MySQL运算符和函数

文章目录 【 1. 算术运算符 】【 2. 逻辑运算符 】2.1 逻辑非 (NOT 或者 !)2.2 逻辑与运算符 (AND 或者 &&)2.3 逻辑或 (OR 或者 ||)2.4 异或运算 (XOR) 【 3. 比较运算符 】3.1 等于 3.2 安全等于运算符 <>3.3 不等于运算符 (<> 或者 !)3.4 小于等于运算符…

Linux基本命令的使用(cp mv)

一、cp命令-1 1、CP命令作用&#xff1a;复制一个源文件到目标文件&#xff08;夹&#xff09; 2、 语法&#xff1a;cp [选项] 源文件 目标文件&#xff08;夹&#xff09; ① 复制到文件夹下&#xff0c;则文件名保持不变 ② 复制到文件中&#xff0c;则文件名变更 二、cp命…