【Java数据结构】深入解析ArrayList与顺序表

news2024/12/23 6:57:37

【Java数据结构】深入解析ArrayList与顺序表

 1.前言~🥳🎉🎉🎉  

2.ArrayList的介绍 

 3.使用ArrayList

3.1 ArrayList的构造方法 

3.11 第一个构造方法

3.12第二个构造方法 

3.13第三个构造方法 

3.2ArrayList中的tostring方法 

3.3ArrayList 的 add 方法 

 4.ArrayList的常用方法

4.1 addAll方法

 4.2subList方法

4.3ArrayList的常用方法总使用 

5.ArrayList的遍历 

 6.总结


 1.前言~🥳🎉🎉🎉  

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

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

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


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

2.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的几个成员变量:



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

 3.使用ArrayList


3.1 ArrayList的构造方法 


3.11 第一个构造方法

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

3.12第二个构造方法 


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

3.13第三个构造方法 


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

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

参数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内部中(按顺序复制)。

3.2ArrayList中的tostring方法 

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

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

3.3ArrayList 的 add 方法 


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

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


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


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


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

 4.ArrayList的常用方法


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

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

4.1 addAll方法

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

boolean addAll(Collection<? extends E> c)

其中有以下要求:

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

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


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

 4.2subList方法

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异常

4.3ArrayList的常用方法总使用 

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

        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);

5.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循环遍历ArrayList
        for (int i = 0; i < arrayList.size(); i++) {
            System.out.print(arrayList.get(i) + " ");
        }
        // 通过for-each循环遍历ArrayList
        for (Integer integer : arrayList) {
            System.out.print(integer + " ");
        }
        // 通过迭代器遍历ArrayList(了解即可,无需深入了解,之后会学)
        Iterator<Integer> it = arrayList.iterator();
        while (it.hasNext()) {
            System.out.print(it.next() + " ");
        }
    }
}

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

 6.总结

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

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

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

相关文章

vue2[黑马笔记]

vue基础 是什么—javascript框架 构建用户界面的前端框架 1.构建用户界面用vue往html页面中填充数据 2.框架现成的解决方案&#xff0c;遵守框架的规范去实现自己的业务功能学习vue 就是学习vue框架中规定的用法vue的指令组件&#xff08;对ul结构的复用&#xff09;&#x…

STM32G431RBT6之LCD与LED配置

首先,配置时钟树,时钟树的配置在我的另外一篇博客里,这里不再赘述. LCD与LED具有共同的IO口,同时创建工程较好. 打开原理图,发现LED的IO口是PC8~PC15,还有一个容易看漏的PD2.LCD的IO口是PC0到PC15. 当然,看产品手册也可以知道,但是还是推荐大家看原理图. 打开cubumx,给PC0~PC…

openjudge_2.5基本算法之搜索_1998:寻找Nemo

题目 1998:寻找Nemo 总时间限制: 2000ms 内存限制: 65536kB 描述 Nemo 是个顽皮的小孩. 一天他一个人跑到深海里去玩. 可是他迷路了. 于是他向父亲 Marlin 发送了求救信号.通过查找地图 Marlin 发现那片海像一个有着墙和门的迷宫.所有的墙都是平行于 X 轴或 Y 轴的. 墙的厚度可…

linux开发板开机启动向日葵

硬件&#xff1a;orangepi 5 pro 操作系统&#xff1a;ubuntu 20.4 lts 安装向日葵 根据我的实测&#xff0c;arm架构的ubuntu系统只能安装向日葵提供的麒麟系统的那个版本&#xff0c;具体安装方式官网下载页面有 允许任意用户连接到 X11 使用root用户登录后打开终端输入一下…

中仕公考:这几类人不能考公务员

拥有以上五类行为的学生无法参加公务员考试&#xff0c;当然这些只是基本的不能报考公务员的类别&#xff0c;现在各地区还会根据自己的情况调整范围&#xff0c;具体要以各地的公告为主。

电脑的无用设置功能(建议关闭)

目录 1、传递优化 ​2、常规​ 3、电源 1、传递优化 2、常规3、电源

【Kylin】V10系统在VMware中分辨率太小,无法通过GUI修改分辨率的解决方法

【Kylin】V10系统在VMware中分辨率太小&#xff0c;无法通过GUI修改分辨率的解决方法 解决办法1.打开终端方法1&#xff1a;方法2 2.输入 xrandr 命令&#xff0c;查询分辨率支持的列表3.选择适合的分辨率 。 例如&#xff1a;xrandr -s 1440x900_60 问题如下图&#xff1a; 保…

重要文件夹怎么加密?文件夹加密软件哪个更好?

文件夹是电脑重要的数据管理工具&#xff0c;可以避免电脑数据混乱。而为了保护文件夹的数据安全&#xff0c;我们需要加密保护重要文件夹。那么&#xff0c;重要文件夹怎么加密呢&#xff1f;下面我们就来了解一下。 文件夹加密超级大师 说起文件夹加密&#xff0c;文件夹加密…

通义千问再开源!推出最大尺寸1100亿参数模型

4月28日消息&#xff0c;通义千问开源1100亿参数模型Qwen1.5-110B&#xff0c;成为全系列首个千亿级参数开源模型&#xff0c;并在多项基准测评中都创下可与Llama3-70B媲美的成绩。目前Qwen1.5系列已累计开源10款大模型&#xff0c;通义千问开源模型下载量超过700万。 通义千问…

【C#】.net core 6.0 MVC返回JsonResult显示API接口返回值不可被JSON反序列化

欢迎来到《小5讲堂》 这是《C#》系列文章&#xff0c;每篇文章将以博主理解的角度展开讲解。 温馨提示&#xff1a;博主能力有限&#xff0c;理解水平有限&#xff0c;若有不对之处望指正&#xff01; 目录 背景API接口接口代码请求失败原因排查调通效果 常见返回类型相关文章 …

python文件 成绩分析

‘’文件score.txt中存储了学生的考试信息,内容如下 小明,98 小刚,90 小红,91 小王,98 小刘,80 请写代码,读取文件数据,并进行如下分析 最高分和最低分分别是多少&#xff1f;得最高分的学生有几个&#xff1f; 得最低分的学生有几个平均分是多少&#xff1f; ‘’’ def rea…

会声会影2024官方中文破解旗舰版下载安装免费激活教程

会声会影2024是一款功能强大的视频编辑软件&#xff0c;它凭借直观易用的界面、全面的编辑工具以及丰富的特效和转场效果&#xff0c;受到了广大视频编辑爱好者和专业人士的喜爱。以下是对会声会影2024的详细介绍&#xff1a; 会声会影最新一键安装包下载如下&#xff1a; htt…

【真实体验】使用崖山YMP 迁移 Oracle/MySQL 至YashanDB 23.2 验证测试【YashanDB迁移体验官】

一、前言 说一下我和崖山数据库的结缘&#xff0c;大概在去年吧&#xff0c;因为我经常在墨天轮写文章&#xff0c;看到崖山数据库推出了一崖山体验官的活动&#xff0c;我就报名参加了。第一次体验了崖山数据库&#xff0c;也测试了我司数据库到崖山数据库的兼容性&#xff0…

Git--分布式版本控制系统

目录 一、理解分布式版本控制系统二、远程仓库三、克隆远程仓库四、向远程仓库推送五、拉取远程仓库六、配置Git七、给命令配置别名八、创建标签九、操作标签 一、理解分布式版本控制系统 我们⽬前所说的所有内容&#xff08;⼯作区&#xff0c;暂存区&#xff0c;版本库等等&a…

【强训笔记】day5

NO.1 思路&#xff1a;找到数量最小的字符&#xff0c;就可以知道you的数量&#xff0c;用o的数量减去you的数量再减去1就是oo的数量。 代码实现&#xff1a; #include<iostream>using namespace std;int main() {int q;cin >> q;int a, b, c;while (q--){cin &g…

2024年SCI新算法独家改进:多策略改进的鹦鹉优化算法!炫技作品!效果极佳!

很多小伙伴经常问我能不能做原创改进的优化算法&#xff0c;这不&#xff0c;今天就给大家发了一期。 经常找我做改进算法的小伙伴&#xff0c;都知道我的实力。每份原创改进算法的代码&#xff0c;都是我阅读了很多文献、实验了很久之后做出来并调试的&#xff0c;不是随随便便…

[Kubernetes] 集群重新初始化

文章目录 1.master节点2.worker节点3.查看结果 1.master节点 kubeadm reset清理信息: cd ~ 进入根目录ll -a 查看是否存在.kube文件 rm -rf /root/.kubesystemctl restart docker ## 重启docker systemctl restart kubelet ## 重启kubeletrm -rf /etc/cni/net.d初始化主节点…

苹果怎么同步备忘录?简单操作大公开

在快节奏的生活中&#xff0c;备忘录成为了我们记录重要事项、管理日常任务不可或缺的工具。苹果设备自带的备忘录应用&#xff0c;不仅界面简洁、操作便捷&#xff0c;还具备强大的同步功能&#xff0c;让我们能够随时随地查看和编辑备忘录内容。那么&#xff0c;苹果怎么同步…

西湖大学赵世钰老师【强化学习的数学原理】学习笔记1节

强化学习的数学原理是由西湖大学赵世钰老师带来的关于RL理论方面的详细课程&#xff0c;本课程深入浅出地介绍了RL的基础原理&#xff0c;前置技能只需要基础的编程能力、概率论以及一部分的高等数学&#xff0c;你听完之后会在大脑里面清晰的勾勒出RL公式推导链条中的每一个部…

qt5-入门-QListWidget-通过右键快捷菜单复制item内容

参考&#xff1a; C GUI Programming with Qt 4, Second Edition 本地环境&#xff1a; win10专业版&#xff0c;64位&#xff0c;Qt5.12 效果 在某个item上右键&#xff0c;点击copy后&#xff0c;item的内容已复制到剪贴板。 实现 #include <QMenu> #include <…