数据结构(Java):顺序表集合类ArrayList

news2025/1/13 17:29:53

1、线性表

线性表,在逻辑结构上是连续的(可理解为连续的一条直线,一对一的关系),而在物理结构上不一定连续,通常以数组和链式结构进行存储。

线性表是一种在实际中广泛使用的数据结构,常见的线性表有:顺序表、链表、栈、队列......

2、顺序表

顺序表是线性表的一种,其物理地址是连续的,采用数组的的存储结构,在数组上完成数据的增删查改。

3、集合类ArrayList

在Java集合框架中,ArrayList是一个泛型类,并且实现了List接口,是Java为我们封装好的顺序表。

当我们想存储哪种类型的元素时,都可以通过泛型传参来构建相应类型的顺序表

接下来,让我为大家仔细讲解一下ArrayList这个集合类。

3.1 ArrayList的成员变量

我们通过观察ArrayList的源码,得到其成员变量如下:

已知,DEFAULT_CAPACITY是指默认容量(10),elementData是用来实际存储数据的数组,size记录顺序表数据的数量,剩下的两个数组均为空数组。

想要知道他们的用途,我们还需要观察ArrayList的构造方法。

3.2 ArrayList的构造方法

通过观察源码,我们依然可以得出其构造方法:

接下来,我们逐个分析。

3.2.1 带参构造之public ArrayList(int initialCapacity)

不难看出,我们传入的参数就是我们初始化顺序表时的容量(即大小)。

通过if-else语句可以看出:

1.传入的参数为正数时,初始化的大小即为参数值。

2.传入的参数为0时,该顺序表数组的引用指向的就是成员变量中的空数组。

3.传入的参数为负数时,则会抛出异常(数组大小肯定为正数)。

public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>(10);//初始容量为10
    }

3.2.2 无参构造之public ArrayList()

这个构造方法就更简单了,无参构造时,该顺序表数组的引用指向的也是成员变量中的空数组。

public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>();
    }

3.2.1 带参构造之public ArrayList(Collection<? extends E> c)

大家看到这个构造方法时,是不是愣住了一下子,哈哈哈~,不要慌张,听我道来。

我们先来看这个参数CollectionCollection是一个接口,不知道大家是否还有印象,它曾出现在集合框架中的顶层:

而<>中的内容,"?"是指通配符(这里了解即可),extends我们在讲解泛型时就已经知道,这规定了通配符的上界为E。

也就是说,该构造方法所传参数必须满足以下几点:

1.实现了Collection接口

2.所具备的泛型必须是E或者E的子类

也就说,我们也可以将另一个顺序表当做这个顺序表构造方法的参数传入。

代码示例:

public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);
        ArrayList<Integer> list1 = new ArrayList<>(list);//将list当做顺序表构造方法的参数传入
        list1.add(99);
        list1.add(100);
        System.out.println(list1);//[1, 2, 3, 99, 100]
    }

3.3 ArrayList的扩容机制

在上面讲顺序表的构造方法时,我们讲当构造方法为无参构造或者传入的参数为零时,数组是空数组。既然是空数组,那么我们如何进行数据的添加或者插入呢?以及当数组容量满时,如何进行数据的添加或者插入呢?

实际上,当无参构造或者传入的参数为零时,经过源码的处理,也是将数组的初始容量设置成了DEFAULT_CAPACITY(10)。

且,ArrayList是一个动态顺序表,即:在插入元素的过程中会以1.5倍自动扩容(调用Arrays的copyOf方法进行扩容)。

3.4 ArrayList的常用方法

ArrayList提供了很多的方法,在这里,我将列举较常用的方法。

3.4.1 add方法(数据插入)

该方法实现了重载,分别在顺序表尾部插入和在指定下标位置插入。

3.4.1.1 尾部插入数据

方法格式:

代码示例:

3.4.1.2 指定位置插入数据

方法格式:

代码示例:

3.4.2 addAll方法

方法格式:

同样的道理,传入的参数必须实现了Collection接口,并且其泛型参数的上界为E(上文已经进行了讲解)。

调用该方法,可以将参数的数据尾插到当前顺序表当中。

代码演示:

3.4.3 remove方法(数据删除)

3.4.3.1 删除指定下标数据

方法格式:

注意:该方法的返回值,为所删除的数据。

3.4.3.2 删除确定数值的数据

方法格式:

因为ArrayList是一个泛型类,数组中的元素都为一个对象,所以我们应该传入相应类型的对象作为参数。

代码示例:

注意:我们看到,remove方法中的参数被划上了横线,这说明该方法的这种传参调用方式被废弃了(不常用了),但是我们仍然可以使用。

3.4.4 get方法(获取数据)

方法格式:

传入下标,返回指定下标的数据。

3.4.5 set方法(修改数据)

方法格式:

修改指定下标的数据。、

3.4.6 set方法(清空顺序表)

方法格式:

 

该方法的底层就是:将有效数据修改为null(各元素为引用类型),再将size(数组大小)置为0。

3.4.7 subList方法(截取顺序表)

方法格式:

该方法能够截取顺序表的指定区间(区间为左闭右开),并返回截取后的顺序表(注意:返回类型为List,是ArrayList实现的一个接口)。

代码示例:

需要注意的,新截取的顺序表list1和原来的顺序表list,指向的的是同一块空间:

也就是说,修改顺序表list1中的数据,list中的数据也会被修改。

3.5 ArrayList的遍历

3.5.1 for循环遍历

代码示例:

public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);
        int size = list.size();
        for (int i = 0; i < size; i++) {
            int data = list.get(i);//得到i下标的元素
            System.out.print(data+" ");
        }
        System.out.println();
    }

3.5.2 for-each循环遍历

因为顺序表实际上是个数组,我们可以通过for-each循环来遍历顺序表:

public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);
        for (int data : list) {
            System.out.print(data+" ");
        }
    }

3.5.3 迭代器遍历

3.5.3.1 Iterator迭代器

在ArrayList中,存在iterator方法:

通过调用这个方法,我们可以得到一个迭代器对象,再通过这个对象将数据一个一个打印,直到全部打印完成。

public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);
        
        Iterator<Integer> iterator =  list.iterator();
        while (iterator.hasNext()) {//hasNext方法 用来判断是否有下一个数据
            int data = iterator.next();//next方法是 得到下一个数据
            System.out.print(data+" ");
        }
    }

hasNext方法 用来判断是否有下一个数据。

next方法是 得到下一个数据。

3.5.3.2 ListIterator迭代器

与Iterator迭代器用法相同,我们可以通过listIterator来遍历顺序表:

public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);

        ListIterator<Integer> listIterator = list.listIterator();
        while (listIterator.hasNext()) {
            int data = listIterator.next();
            System.out.print(data+" ");
        }
    }
3.5.3.3  Iterator迭代器和ListIterator迭代器的关系

首先,这两者均为接口

其次,我们通过源码可以发现,ListIterator拓展了Iterator,也就是说,ListIterator迭代器拓展了Iterator迭代器的功能,其方法更加丰富。

3.5.3.3.1 ListIterator迭代器的新增功能

这里,我们演示一个ListIterator的新增方法。

通过源码,我们可以发现ArrayList的listIterator方法可以传入参数,

调用listIterator方法并传入参数,得到的迭代器就会来到该下标的元素之后的位置,

我们再使用ListIterator新增的hasPrevious和previous方法就可以实现对顺序表从后往前的遍历:

代码:

public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);

        int size = list.size();
        ListIterator<Integer> listIterator = list.listIterator(size);
        while (listIterator.hasPrevious()) {
            int data = listIterator.previous();
            System.out.print(data+" ");
        }
    }

OK~本次博客到这里就结束了,

感谢大家的阅读~欢迎大家在评论区交流问题~

如果博客出现错误可以提在评论区~

创作不易,请大家多多支持~

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

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

相关文章

【最佳实践】 使用react写一个sliderbar 侧滑导航组件

大家好&#xff0c;我是DX3906 我们现在使用React 来实现一个侧边导航组件&#xff08;sidebar&#xff09;&#xff0c;你可以使用 CSS 和 React 的状态管理来实现。下面是一个简单的示例&#xff0c;展示如何创建一个基本的侧边导航组件&#xff1a; 1. 安装依赖&#xff1a;…

浅谈逻辑控制器之Include控制器

浅谈逻辑控制器之Include控制器 “Include控制器”是一个特别实用的功能&#xff0c;它允许用户将一个或多个测试片段&#xff08;通常是请求&#xff09;包含到不同的测试计划或模块中&#xff0c;从而实现代码的复用和测试结构的优化。本文档将详细介绍JMeter中的Include控制…

并发编程-02深入理解Java线程

一 线程基础知识 1.1 理解线程和进程 进程和线程的概念: 进程: 程序由指令和数据组成&#xff0c;但这些指令要运行&#xff0c;数据要读写&#xff0c;就必须将指令加载至CPU&#xff0c;数据加载至内存。在指令运行过程中还需要用到磁盘、网络等设备。进程就是用来加载指…

Gobject tutorial 十

参考&#xff1a;GLib – 2.0: The Main Event Loop The Main Event Loop 主事件循环管理所有可用的事件源&#xff0c;事件可以是各种类型、各种数量的。比如说文件描述符(普通文件、管道以及套接字)和超时。 新类型的事件源可以通过函数g_source_attach来添加。为了使多个…

输入/输出文字

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 在海龟绘图中&#xff0c;也可以输入或者输出文字&#xff0c;下面分别进行介绍。 1 输出文字 输出文字可以使用write()方法来实现&#xff0c;语…

【QCustomPlot实战系列】QCPGraph区域高亮

使用QCPDataSelection来设置选中的区域&#xff0c;并将QCPGraph的可选择区域设置成QCP::stMultipleDataRanges void AreaPieces::initCustomPlot(QCustomPlot *parentPlot) {QVector<double> x {0, 1, 2, 3, 4, 5, 6, 7, 8};QVector<double> y {200, 560, 750…

解决Vue+Vite打包后Leaflet的marker图标不显示的问题

前言 用Leaflet写关于WebGIS的开发&#xff0c;用Vite或者webpack打包&#xff0c;打包后会找不到图标&#xff0c;如下所示。 直言的说&#xff0c;笔者去网上搜了搜&#xff0c;其实收到一个比较好是答案。网址如下。 &#xff08;完美解决~&#xff09;关于VueLeaflet添加…

P2实验室装修标准都有哪些

P2实验室&#xff08;也称为生物安全二级实验室&#xff0c;BSL-2实验室&#xff09;的装修标准需要满足一系列的设计和施工要求&#xff0c;以确保实验室的安全性和功能性。因此&#xff0c;P2实验室装修标准不仅要满足一般实验室的要求&#xff0c;还需符合生物安全的特殊规定…

企业运维六边形战士 质量稳定 效率为王

随着信息化的不断深入和扩展&#xff0c;企业IT系统的复杂性和设备多样性日益增加。为了保障业务的高可用性和连续性&#xff0c;企业需要一个全面、高效、智能的一体化运维管理平台。在用户市场的推动下&#xff0c;LinkSLA智能运维管家展现出【六边形战士】的优质属性&#x…

数据结构-----【链表:刷题】

-------------------------------------------基础题参照leetcode---------------------------------------------------------------------------------------------------------- 【2】两数相加 /*** Definition for singly-linked list.* struct ListNode {* int val;…

ChatBI开源实现: 基于SuperSonic的AI+BI的产品设计

产品起源 为什么要做这样的产品&#xff1f;文章《ChatBI开源实现: AIBI的产品设计》中有介绍 为什么要自己做这样的产品&#xff1f;1、低成本试错&#xff1b;2、未来数据生态入口&#xff1b; 为什么要基于Supersonic做&#xff1f; 开源协议友好&#xff1a;可魔改商用 社区…

仿真分析 + AI:创建基础设施的新未来

吴付标 大漠风电&#xff0c;深远海平台&#xff0c;一带一路上的高铁、电站…… 在新一轮的基础设施建设浪潮中&#xff0c;项目与结构变得越来越复杂&#xff0c;碳中和与可持续发展的要求越来越高&#xff0c;仿真分析技术应运而兴。 在最近召开的Bentley软件2024创新智旅 …

【Python/Pytorch - 网络模型】-- 高阶SVD算法

文章目录 文章目录 00 写在前面01 基于Python版本的高阶SVD算代码02 HOSVD 的步骤 00 写在前面 高阶奇异值分解&#xff08;Higher-Order SVD&#xff0c;HOSVD&#xff09;是一种将传统的奇异值分解&#xff08;SVD&#xff09;扩展到高阶张量的方法。它能够将一个高阶张量分…

FileNotFoundError: Cannot find DGL C++ graphbolt library at ...

FileNotFoundError: Cannot find DGL C graphbolt library at ...-CSDN博客https://blog.csdn.net/weixin_44017989/article/details/137658749

SAP ABAP 之OOALV

文章目录 前言一、案例介绍/笔者需求二、SE24 查看类 a.基本属性 Properties b.接口 Interfaces c.友元 Friends d.属性 Attributes e.方法 Methods f.事件 Events g.局部类型 Types …

韩顺平0基础学java——第31天

p612-637 IO流 IO流原理及流的分类 Java lO流原理 1.I/O是Input/Output的缩弓&#xff0c;IV/O技术是非常实用的技术&#xff0c;用于处理数据传输。 如读/写文件&#xff0c;网络通讯等。 2. Java程序中&#xff0c;对于数据的输入/输出操作以”流(stream)”的方式进行。 3…

Orangepi Zero2 全志H616 的初识

目录 一、全志H616简介 1.1 为什么学&#xff1a; 1.2 学什么&#xff1a; 1.3 全志H616平台介绍&#xff1a; 二、刷机 系统启动和初始化配置 2.1 需要的工具&#xff1a; 2.2 工具安装&#xff1a; 2.3 刷机&#xff1a; 2.4 登录系统&#xff1a; 2.5 修改登录密码…

Android开发系列(十)Jetpack Compose之Card

Card是一种常用的UI组件&#xff0c;用于显示一个具有卡片样式的容器。Card组件通常用于显示列表项、卡片式布局或任何需要显示边框和阴影的UI元素。 使用Card组件&#xff0c;您可以轻松地创建带有卡片效果的UI元素。以下是一些Card组件的常见属性和功能&#xff1a; elevati…

云计算基础知识

前言&#xff1a; 随着ICT技术的高速发展&#xff0c;企业架构对计算、存储、网络资源的需求更高&#xff0c;急需一种新的架构来承载业务&#xff0c;以获得持续&#xff0c;高速&#xff0c;高效的发展&#xff0c;云计算应运而生。 云计算背景 信息大爆炸时代&#xff1a…