ArrayList与顺序表(2)

 前言~🥳🎉🎉🎉  

hellohello~,大家好💕💕,这里是E绵绵呀✋✋ ,如果觉得这篇文章还不错的话还请点赞❤️❤️收藏💞 💞 关注💥💥,如果发现这篇文章有问题的话,欢迎各位评论留言指正,大家一起加油!一起chin up!👍👍 

💥个人主页:E绵绵的博客
💥所属专栏:JAVA知识点专栏   JAVA题目练习  c语言知识点专栏   c语言题目练习

在上一章我们将顺序表的模拟讲了之后,我们现在正式开始介绍ArrayList这个类(顺序表)。开始吧! 


参考文章:【Java 数据结构】顺序表_数据结构java顺序表基本算法测试-CSDN博客

ArrayList的介绍 

 在集合框架中,ArrayList是一个普通的类,实现了List接口,具体框架图如下:

❤️❤️由这可知:

1. ArrayList是以泛型方式实现的,所以使用时必须要先实例化

2. ArrayList实现了RandomAccess接口,表明ArrayList支持随机访问

3. ArrayList实现了Cloneable接口,表明ArrayList是可以clone的

4. ArrayList实现了Serializable接口,表明ArrayList是支持序列化的

5. 和Vector不同,ArrayList不是线程安全的,在单线程下可以使用,在多线程中可以选择Vector或者 CopyOnWriteArrayList

6. ArrayList底层是一段连续的空间,并且可以动态扩容,是一个动态类型的顺序表


我们查看源码可知的确实施了这些接口。

这就是类定义的前部分,这里还是比较复杂的,会随着我们学习的深入,逐步学习到。

❤️❤️接下来我们来看ArrayList的几个成员变量:



对于第一个成员变量,我们无需了解,只需要关注下后面五个成员变量。

 使用ArrayList


 ArrayList的构造方法 


 第一个构造方法

当前是一个带参数的构造方法,很好理解,根据传递的参数开辟数组空间的大小。如果参数是等于0,就直接把 EMPTY_ELEMENTDATA 这个空数组赋值给存放数据的数组中。 如果是给定一个负数,显然是错误的,也即直接抛出异常!

第二个构造方法 


对于这个无参构造方法居然也是给了一个空数组,也就是没有分配数组内存,那它到底是怎么把数据放进去的?别急,随着后面的讲解,我们会解开这个谜题。

第三个构造方法 

 


里面涉及了泛型的进阶,我们这也不怎么好描述,就直接说特点吧:

下面是这个构造方法的一些特点:

参数c要求是实现了Collection接口的对象,

参数c中的<>元素类型必须与ArrayList中的<>元素类型兼容,即参数c中的<>元素类型必须是ArrayList中<>元素类型的子类或者相同类型。

在使用该构造方法后,它会按照参数c中元素的顺序将元素添加到新创建的ArrayList中。

实例如下:

public class Test {public static void main(String[] args) {List<String> list = new ArrayList<>();list.add("apple");list.add("banana");list.add("orange");ArrayList<String> arrayList = new ArrayList<>(list);System.out.println(arrayList);}}


所以由这可知我们就将list内部的元素apple banana orange全部复制到Arraylist内部中(按顺序复制)。

ArrayList中的tostring方法 

由上图可知我们的ArrayList重写了toString方法。所以在用println时,内部参数为ArrayList对象时其将打印出该对象内部的所有元素,以字符串形式打印出来。

例如,如果ArrayList中有三个元素 “apple”、“banana” 和 “orange”,那么调用println方法将打印出字符串 “[apple, banana, orange]”。

ArrayList 的 add 方法 


别小看这几行代码,跟我们自己模拟实现的还是有区别的,真正有内涵的代码其实在 ensureCapacityInternal 这个方法中,那么现在,我们就一步步去解开他的面纱:

由这可知,我们就可以解开之前构造方法中的问题:我们用无参构造方法给其elementData一个空数组,也就是没有分配内存,那么它到底怎么存放数据进去?


我们看源码可知如果在使用add时elementData指向一个空数组,那么在使用add方法时内部的ensureCapacityInternal方法会给其element重新指向一个可以存放十个数据的数组,这样就可以存放数据了。


我们还从源码处可知在使用add时如果其数组中的数据已经满了,那么其会为该数组扩容1.5倍再存放数据。


所以这就是add的厉害之处。不仅能给内部为空数组的ArrayList对象重新开辟一个内部为10个数据的数组,使其能存放数据; 它还可以为满数据的数组实现扩容,使其也能存放数据。

 ArrayList的常用方法


在这里我们就重点讲两个方法 addAll和subList,

❤️❤️对于其他方法的使用,都很简单,自己去查源码,我这就不讲了。到了数据结构阶段,就要尝试着自己看源码,培养自主学习的能力!

 addAll方法

ArrayList中的addAll方法是用于将另一个集合c中的所有元素添加到当前ArrayList对象中的方法。它的语法如下:

boolean addAll(Collection<? extends E> c)

其中有以下要求:

参数c要求是实现了Collection接口的对象,

参数c中的<>元素类型必须与ArrayList中的<>元素类型兼容,即参数c中的<>元素类型必须是ArrayList中<>元素类型的子类或者相同类型。


这个方法可以用于批量添加元素到ArrayList对象中,方便快捷。注意,addAll方法不会去重,如果添加的元素在当前ArrayList中已经存在,则会重复添加

 subList方法

ArrayList中的subList方法用于获取原顺序表的一个子顺序表。它接受两个参数,分别是起始索引(fromindex)和结束索引(toindex),其左闭右开。一个新的List对象,包含原顺序表中指定范围内的元素。


subList方法返回的子顺序表在原顺序表的内部,对子顺序表的修改会反映到原顺序表上,反之亦然。这意味着,如果你修改了子顺序表中的元素,原顺序表也会相应地被修改;如果你修改了原顺序表中的元素,子顺序表也会相应地被修改。


需要注意的是,如果有一个顺序表此时存在一个子顺序表,现在将该顺序表结构性修改(如添加或删除元素),之后再用println打印子顺序表,会导致ConcurrentModificationException异常抛出。这是因为结构性修改改变了原顺序表的大小,从而也破坏了其子列表的大小,所以打印子列表时就报错。(内容牵涉到了迭代器,这里就不细讲)


ArrayList<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
list.add(3);
list.add(4);
list.add(5);List<Integer> subList = list.subList(1, 4);
System.out.println(subList);  // 输出:[2, 3, 4]subList.set(0, 10);
System.out.println(list);  // 输出:[1, 10, 3, 4, 5]list.add(6);
System.out.println(subList);  // 抛出ConcurrentModificationException异常

ArrayList的常用方法总使用 

 ❤️❤️以下是我们对常用方法的总使用,大家了解一下。

        ArrayList<Integer> arrayList = new ArrayList<>();// 1,插入(尾插)arrayList.add(1);arrayList.add(2);arrayList.add(3);arrayList.add(4);arrayList.add(5);System.out.println("插入数据后:" + arrayList);// 2,在任意位置插(下标)插入arrayList.add(0, -1);System.out.println("在0下标插入-1后:" + arrayList);// new 一个链表对象并尾插“1”,“2”LinkedList<Integer> linkedList = new LinkedList<>();linkedList.addLast(1);linkedList.addLast(2);// 3,插入 linklist 的所有元素arrayList.addAll(linkedList);System.out.println("插入linklist后:" + arrayList);// 4,删除任意位置(下标)数据arrayList.remove(0);System.out.println("删除0下标数据后:" + arrayList);// 5,删除指定数据arrayList.remove(new Integer(1));// 参数为Object(类)类型的对象System.out.println("删除第一个1后:" + arrayList);// 上面两个remove()方法构成了重载// remove()方法的参数如果输入整数之后不会自动装箱,而是会被自动识别为index。// 因为index是int类型,而顺序表中的数据是Object(类)类型// 但在插入数据时,输入的是1,基本类型,它会自动装箱,变成Integer类型//删除数据时之所以输入整数时不自动装箱是因为其参数本身就有index(整数)这个类型//所以优先选择该整型参数,不会自动装箱。// 所以要删除数据,应该输入类类型的对象,而不是基本类型的数据// 6,获取任意位置(下标)数据int ret = arrayList.get(0);System.out.println("得到0下标的数据:" + ret);// 7,更改任意位置(下标)数据arrayList.set(0,100);System.out.println("把0下标数据改成100后:" + arrayList);// 8,判断是否存在该数据boolean bl = arrayList.contains(100);System.out.println("判断是否存在100这个数据:" + bl);// 9,返回第一个key的位置(下标)int index = arrayList.indexOf(2);System.out.println("第一个2的下标:" + index);// 10,返回最后一个key的位置(下标)int lastIndex = arrayList.lastIndexOf(2);System.out.println("最后一个2的下标:" + lastIndex);// 11,获取顺序表长度int size = arrayList.size();System.out.println("顺序表长度为:" + size);// 12,截取List<Integer> list  = arrayList.subList(1, 3);// [1,3)左闭右开,返回出的子顺序表和ArrayList共用一个elementData数组System.out.println(list);// 13,清空顺序表arrayList.clear();System.out.println("清空顺序表后:" + arrayList);

ArrayList的遍历 

 ❤️❤️对于顺序表的遍历,我们可以通过 for 循环,for-each,以及迭代器的方法遍历:

public class TestArrayList {public static void main(String[] args) {ArrayList<Integer> arrayList = new ArrayList<>();arrayList.add(1);arrayList.add(2);arrayList.add(3);// 通过for循环遍历ArrayListfor (int i = 0; i < arrayList.size(); i++) {System.out.print(arrayList.get(i) + " ");}// 通过for-each循环遍历ArrayListfor (Integer integer : arrayList) {System.out.print(integer + " ");}// 通过迭代器遍历ArrayList(了解即可,无需深入了解,之后会学)Iterator<Integer> it = arrayList.iterator();while (it.hasNext()) {System.out.print(it.next() + " ");}}
}

除此以外,我们还可以通过println(ArrayList对象)打印出该对象内部的所有元素,这个方法我们在之前就讲过了,这里不多讲述了。

 总结

所以对于我们的顺序表的相关知识点就讲完啦,之后将给大家介绍链表,敬请期待! 还希望各位大佬们能给个三连,点点关注,点点赞,发发评论呀,感谢各位大佬~❤️❤️💕💕🥳🎉🎉🎉

 

 

 

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

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

相关文章

【THM】Linux Privilege Escalation(权限提升)-初级渗透测试

介绍 权限升级是一个旅程。没有灵丹妙药,很大程度上取决于目标系统的具体配置。内核版本、安装的应用程序、支持的编程语言、其他用户的密码是影响您通往 root shell 之路的几个关键要素。 该房间旨在涵盖主要的权限升级向量,并让您更好地了解该过程。无论您是参加 CTF、参加…

海睿思受邀参加 “走进海澜之家”研习交流,共探制造业数据治理的新模式

近日&#xff0c;OceanMind海睿思参加了由江苏省企业信息化协会&#xff08;以下简称“苏信会”&#xff09;主办的“走进海澜之家集团股份有限公司”研习交流活动。 海睿思与南京我乐家居、江苏今世缘酒业、江苏鱼跃医疗等近30家制造业领域的优秀企业代表&#xff0c;一同围绕…

【精简改造版】大型多人在线游戏BrowserQuest服务器Golang框架解析(2)——服务端架构

1.架构选型 B/S架构&#xff1a;支持PC、平板、手机等多个平台 2.技术选型 &#xff08;1&#xff09;客户端web技术&#xff1a; HTML5 Canvas&#xff1a;支持基于2D平铺的图形引擎 Web workers&#xff1a;允许在不减慢主页UI的情况下初始化大型世界地图。 localStorag…

为何申请图形商标的通过率比文字商标低!

最近有朋友找普推知产老杨说要申请图形商标&#xff0c;从全国的平均通过率来看&#xff0c;图形商标的通过率要比文字低一些&#xff0c;因为图形商标存在许多不确定因素。 文字在检索时只要不在盲查期&#xff0c;基本都可以搜索出来&#xff0c;是相同还是高近差不多就可以检…

符号内链接文件作用域:static限制在本模块

示例&#xff1a; /*** brief how about symbol-inner? show you here.* author wenxuanpei* email 15873152445163.com(query for any question here)*/ #define _CRT_SECURE_NO_WARNINGS//support c-library in Microsoft-Visual-Studio #include <stdio.h>static…

基于K-means和FCM算法的合成纹理图像及SAR图像的分割

&#x1f380;个人主页&#xff1a; https://zhangxiaoshu.blog.csdn.net &#x1f4e2;欢迎大家&#xff1a;关注&#x1f50d;点赞&#x1f44d;评论&#x1f4dd;收藏⭐️&#xff0c;如有错误敬请指正! &#x1f495;未来很长&#xff0c;值得我们全力奔赴更美好的生活&…

使用Docker搭建本地Nexus私有仓库

0-1开始Java语言编程之路 一、Ubuntu下Java语言环境搭建 二、Ubuntu下Docker环境安装 三、使用Docker搭建本地Nexus Maven私有仓库 四、Ubuntu下使用VisualStudioCode进行Java开发 你需要Nexus Java应用编译构建的一种主流方式就是通过Maven, Maven可以很方便的管理Java应用的…

wsl2 中docker安装完毕后无法正常启动

wsl2 中docker安装完毕后无法正常启动 1、背景2、目标3、环境4、原因4、操作5.1、查看配置5.2、 切换配置5.3、启动docker5.4、验证docker 1、背景 在win10中安装wsl2体验linux操作系统&#xff0c;按照docker官网步骤安装&#xff0c;安装完毕后面提示 $ docker ps Cannot co…

每日一题 — 二分查找

704. 二分查找 - 力扣&#xff08;LeetCode&#xff09; 朴素二分查找模板&#xff1a; while(.......){//防止溢出int mid left(right - left)/2;if(........){right mid-1;}else if(......){left mid1;}else{return mid;}} 代码&#xff1a; public int search(int[] num…

【前端】项目中如何快速构建vue脚手架

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、vue脚手架介绍二、构建vue脚手架三、总结 前言 随着开发语言的普及&#xff0c;越来越多的开发框架需要搭建ue脚手架来进行敏捷开发&#xff0c;本文主要介…

软考高项(已通过,E类人才)-学习笔记材料梳理汇总

软考高项&#xff0c;即软考高级信息系统项目管理师&#xff0c;全国计算机技术与软件专业技术资格&#xff08;水平&#xff09;考试中的高级水平测试。适用于从事计算机应用技术、软件、网络、信息系统和信息服务等领域的专业人员&#xff0c;以及各级企业管理人员和从事项目…

MySQL学习笔记5——函数和索引

函数和索引 一、函数1、数学函数2、字符串函数3、条件判断函数 二、索引1、索引是什么2、单字段索引3、组合索引4、总结 一、函数 MySQL提供了很多功能强大&#xff0c;而且使用起来非常方便的函数&#xff0c;包括数学函数、字符串处理函数和条件判断函数等。 1、数学函数 …

Spring Cloud面试篇

面试篇-nacos面试题 1. springboot常见组件 注册中心组件&#xff1a;Eureka、Nacos 负载均衡组件&#xff1a;Ribbon 远程调用组件&#xff1a;OpenFeign 网关组件&#xff1a;Zuul、Gateway 服务保护组件&#xff1a;Hystrix、Sentinel 服务配置管理组件&#xff1a;SpringCl…

大模型的研究新方向:混合专家模型(MoE)

大模型的发展已经到了一个瓶颈期,包括被业内所诟病的罔顾事实而产生的“幻觉”问题、深层次的逻辑理解能力、数学推理能力等,想要解决这些问题就不得不继续增加模型的复杂度。随着不同应用场景的实际需求,大模型的参数会变得越来越大,复杂性和规模不断的增加,尤其是在多模…

18.基础乐理-音阶是什么、有什么用

音阶&#xff1a; 将调式中的音&#xff0c;从以主音开始到以主音结束&#xff0c;由低到高&#xff08;叫做上行&#xff08;xing&#xff09;&#xff09;&#xff0c;或由高到低&#xff08;叫做下行&#xff08;xing&#xff09;&#xff09;&#xff0c;以阶梯状排列起来&…

Bingbong的回文路径

Here 利用回文串&#xff0c;从左往右与从右往左的hash值相同来判断从左往右&#xff0c;例&#xff1a;从右往左&#xff0c;例&#xff1a;由于在树上&#xff0c;考虑建两颗树&#xff0c;一颗根为最高位&#xff08;up&#xff09;&#xff0c;一棵根为最低位&#xff08;…

Rust 使用结构体组织相关联的数据

目录 结构体的定义和实例化 使用字段初始化简写语法使用结构体更新语法从其他实例创建实例使用没有命名字段的元组结构体来创建不同的类型没有任何字段的类单元结构体结构体示例程序 通过派生 trait 增加实用功能方法语法 定义方法带有更多参数的方法关联函数多个 impl 块本文有…

大厂常见算法50题-反转链表

专栏持续更新50道算法题&#xff0c;都是大厂高频算法题&#xff0c;建议关注。 文章目录 解法参考链接题目解法一 双指针解法二 递归解法三 妖魔化的双指针总结 解法参考链接 题目 解法一 双指针 定义两个指针&#xff1a; pre 和 cur。pre 在前 cur 在后。每次让 pre的 nex…

Day4 商品管理

Day4 商品管理 这里会总结构建项目过程中遇到的问题&#xff0c;以及一些个人思考&#xff01;&#xff01; 学习方法&#xff1a; 1 github源码 文档 官网 2 内容复现 &#xff0c;实际操作 项目源码同步更新到github 欢迎大家star~ 后期会更新并上传前端项目 编写品牌服务 …

在线预约订房酒店小程序源码系统 带完整的安装代码包以及=安装部署教程

传统的酒店预订方式往往依赖于电话、邮件或者到店咨询&#xff0c;这种方式不仅效率低下&#xff0c;而且容易造成信息不准确、沟通不畅等问题。随着智能手机的普及和移动互联网的发展&#xff0c;用户对于随时随地、方便快捷地进行酒店预订的需求日益增强。小编给大家分享一款…