教你如何高效使用Java中的ArrayList

哈喽,各位小伙伴们,你们好呀,我是喵手。运营社区:C站/掘金/腾讯云;欢迎大家常来逛逛

  今天我要给大家分享一些自己日常学习到的一些知识点,并以文字的形式跟大家一起交流,互相学习,一个人虽可以走的更快,但一群人可以走的更远。

  我是一名后端开发爱好者,工作日常接触到最多的就是Java语言啦,所以我都尽量抽业余时间把自己所学到所会的,通过文章的形式进行输出,希望以这种方式帮助到更多的初学者或者想入门的小伙伴们,同时也能对自己的技术进行沉淀,加以复盘,查缺补漏。

小伙伴们在批阅的过程中,如果觉得文章不错,欢迎点赞、收藏、关注哦。三连即是对作者我写作道路上最好的鼓励与支持!

前言

  在Java编程中,集合是非常重要的一个概念。它是一种包装多个对象的数据结构,通常能够动态地增加或删除元素,并能够方便地访问其中的元素。其中,ArrayList就是一个非常常用的集合类,本文将对其进行详细解析。

摘要

  本文将从源代码解析、应用场景案例、优缺点分析、类代码方法介绍、测试用例等方面,深入剖析ArrayList的实现原理和使用方法,旨在帮助读者更好地理解和使用该集合类。

ArrayList集合

简介

  ArrayList是Java集合框架中的一个动态数组,它继承了AbstractList类并实现了List接口,可以存储任意类型的对象。在添加元素时,ArrayList会自动扩容,因此我们可以直接通过下标访问其中的元素。ArrayList还支持在任意位置的插入和删除操作,因此它可以非常方便地使用。

源代码解析

  ArrayList的源代码非常长,这里只对其中一些比较重要的方法进行分析。

add方法

public boolean add(E e) {
    ensureCapacityInternal(size + 1);  // Increments modCount!!
    elementData[size++] = e;
    return true;
}

  该方法用于向ArrayList中添加元素。首先,它会调用ensureCapacityInternal方法,该方法用于确保ArrayList内部数组的容量足够,如果不够则进行扩容。然后,它会将元素添加到数组的尾部,并返回true表示插入成功。

remove方法

public E remove(int index) {
    rangeCheck(index);

    modCount++;
    E oldValue = elementData(index);

    int numMoved = size - index - 1;
    if (numMoved > 0)
        System.arraycopy(elementData, index+1, elementData, index,
                         numMoved);
    elementData[--size] = null; // clear to let GC do its work

    return oldValue;
}

  该方法用于从ArrayList中删除指定位置的元素。首先,它会调用rangeCheck方法,该方法用于检查指定的下标是否越界。然后,它会将元素从数组中删除,并返回该元素的值。需要注意的是,删除元素后,ArrayList内部数组的大小会自动减小,并将被删元素所占据的空间赋为null,以便由垃圾回收器回收。

iterator方法

public Iterator<E> iterator() {
    return new Itr();
}

private class Itr implements Iterator<E> {
    int cursor;       // index of next element to return
    int lastRet = -1; // index of last element returned; -1 if no such
    int expectedModCount = modCount;

    public boolean hasNext() {
        return cursor != size;
    }

    @SuppressWarnings("unchecked")
    public E next() {
        checkForComodification();
        int i = cursor;
        if (i >= size)
            throw new NoSuchElementException();
        Object[] elementData = ArrayList.this.elementData;
        if (i >= elementData.length)
            throw new ConcurrentModificationException();
        cursor = i + 1;
        return (E) elementData[lastRet = i];
    }

    public void remove() {
        if (lastRet < 0)
            throw new IllegalStateException();
        checkForComodification();

        try {
            ArrayList.this.remove(lastRet);
            cursor = lastRet;
            lastRet = -1;
            expectedModCount = modCount;
        } catch (IndexOutOfBoundsException ex) {
            throw new ConcurrentModificationException();
        }
    }

    final void checkForComodification() {
        if (modCount != expectedModCount)
            throw new ConcurrentModificationException();
    }
}

  该方法返回一个Iterator对象,用于遍历ArrayList的元素。这里采用了内部类的形式,定义了一个Itr类,实现了Iterator<E>接口。该类维护了一个游标cursor,用于记录遍历的位置。在next方法中,它会检查是否有其他线程对ArrayList进行修改,如果有则抛出ConcurrentModificationException异常。如果没有,则返回下一个元素并将游标+1。

ensureCapacityInternal方法

private void ensureCapacityInternal(int minCapacity) {
    ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}

private static int calculateCapacity(Object[] elementData, int minCapacity) {
    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        return Math.max(DEFAULT_CAPACITY, minCapacity);
    }
    return minCapacity;
}

private void ensureExplicitCapacity(int minCapacity) {
    modCount++;

    // overflow-conscious code
    if (minCapacity - elementData.length > 0)
        grow(minCapacity);
}

  该方法用于确保ArrayList内部数组的容量足够。首先,它会调用calculateCapacity方法计算需要的最小容量。如果ArrayList内部数组为空,它会将最小容量与默认容量DEFAULT_CAPACITY进行比较,取较大值作为最终容量。否则,它直接返回最小容量。接着,它会调用ensureExplicitCapacity方法,该方法用于在必要时进行扩容。

grow方法

private void grow(int minCapacity) {
    // overflow-conscious code
    int oldCapacity = elementData.length;
    int newCapacity = oldCapacity + (oldCapacity >> 1);
    if (newCapacity - minCapacity < 0)
        newCapacity = minCapacity;
    if (newCapacity - MAX_ARRAY_SIZE > 0)
        newCapacity = hugeCapacity(minCapacity);
    // minCapacity is usually close to size, so this is a win:
    elementData = Arrays.copyOf(elementData, newCapacity);
}

  该方法用于对ArrayList内部数组进行扩容。首先,它会计算出新的容量newCapacity。这里采用了位运算的方法,将原来的容量右移一位,然后与原来的容量进行相加,得到新的容量。接着,它会将新容量与最小容量进行比较,并将较大者作为新容量。如果新容量超过了MAX_ARRAY_SIZE,它会调用hugeCapacity方法进行处理。最后,它会调用Arrays.copyOf方法实现数组的扩容。

其他方法

  除了上述几个重要的方法,ArrayList还包括很多其他方法,如get、set、isEmpty、size等等,这里不一一赘述。

如下是部分源码截图,如若想学习更多,同学们可以尽情前往。

在这里插入图片描述

应用场景案例

  ArrayList在Java编程中的应用非常广泛,下面列举几个常见的应用场景:

  1. 需要动态地添加或删除元素,且需要支持随机访问的情况下,可以使用ArrayList。

  2. 需要对一个已知集合进行遍历或查找时,可以将该集合存储在ArrayList中。

  3. 需要对一个元素集合进行排序时,可以将其存储在ArrayList中,并使用Collections.sort方法进行排序。

优缺点分析

优点

  1. 支持动态扩容,因此可以根据需要动态地添加或删除元素。

  2. 可以支持随机访问,因此可以直接通过下标访问指定位置的元素,效率较高。

  3. ArrayList是一个线程不安全的类,因此在单线程环境下,其操作效率非常高。

缺点

  1. 删除元素时,需要将元素所在的位置之后的所有元素向前移动一位,效率较低。

  2. 插入元素时,可能需要进行数组复制和元素移动的操作,效率也较低。

  3. ArrayList是一个线程不安全的类,因此在多线程环境下,需要采取额外的措施保证线程安全。

类代码方法介绍

ArrayList类包含了很多方法,其中一些比较重要的方法介绍如下:

构造函数

public ArrayList() {
    this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}

public ArrayList(int initialCapacity) {
    if (initialCapacity > 0) {
        this.elementData = new Object[initialCapacity];
    } else if (initialCapacity == 0) {
        this.elementData = EMPTY_ELEMENTDATA;
    } else {
        throw new IllegalArgumentException("Illegal Capacity: "+
                                               initialCapacity);
    }
}

public ArrayList(Collection<? extends E> c) {
    elementData = c.toArray();
    if ((size = elementData.length) != 0) {
        // c.toArray might (incorrectly) not return Object[] (see 6260652)
        if (elementData.getClass() != Object[].class)
            elementData = Arrays.copyOf(elementData, size, Object[].class);
    } else {
        // replace with empty array.
        this.elementData = EMPTY_ELEMENTDATA;
    }
}

  ArrayList类有三个构造函数,分别是无参构造函数、指定初始容量的构造函数和从其他集合类中构造的构造函数。其中,无参构造函数将elementData数组初始化为DEFAULTCAPACITY_EMPTY_ELEMENTDATA,即一个空数组。指定初始容量的构造函数将elementData数组初始化为指定大小的数组。从其他集合类中构造的构造函数将elementData数组初始化为其他集合类的转换结果。

add方法

public boolean add(E e) {
    ensureCapacityInternal(size + 1);  // Increments modCount!!
    elementData[size++] = e;
    return true;
}

public void add(int index, E element) {
    rangeCheckForAdd(index);

    ensureCapacityInternal(size + 1);  // Increments modCount!!
    System.arraycopy(elementData, index, elementData, index + 1,
                     size - index);
    elementData[index] = element;
    size++;
}

  ArrayList类提供了两个方法用于添加元素,分别是在末尾添加元素和在指定位置添加元素。在末尾添加元素时,ArrayList会自动扩容,然后将元素添加到数组的末尾。在指定位置添加元素时,需要调用System.arraycopy方法将插入位置之后的元素向后移动一位,然后将元素插入到指定位置,并调整size属性的值。

remove方法

public E remove(int index) {
    rangeCheck(index);

    modCount++;
    E oldValue = elementData(index);

    int numMoved = size - index - 1;
    if (numMoved > 0)
        System.arraycopy(elementData, index+1, elementData, index,
                         numMoved);
    elementData[--size] = null; // clear to let GC do its work

    return oldValue;
}

public boolean remove(Object o) {
    if (o == null) {
        for (int index = 0; index < size; index++)
            if (elementData[index] == null) {
                fastRemove(index);
                return true;
            }
    } else {
        for (int index = 0; index < size; index++)
            if (o.equals(elementData[index])) {
                fastRemove(index);
                return true;
            }
    }
    return false;
}

private void fastRemove(int index) {
    modCount++;
    int numMoved = size - index - 1;
    if (numMoved > 0)
        System.arraycopy(elementData, index+1, elementData, index,
                         numMoved);
    elementData[--size] = null; // clear to let GC do its work
}

  ArrayList类提供了两个方法用于删除元素,分别是按照下标删除元素和按照元素值删除元素。按照下标删除元素时,需要将删除位置之后的元素向前移动一位,并将删除位置所在的元素赋为null,最后调整size属性的值。按照元素值删除元素时,需要遍历整个数组,找到要删除的元素所在的位置,然后调用fastRemove方法进行删除。

set方法

public E set(int index, E element) {
    rangeCheck(index);

    E oldValue = elementData(index);
    elementData[index] = element;
    return oldValue;
}

  ArrayList类提供了set方法用于替换指定位置的元素。该方法首先需要检查指定位置是否越界,然后将指定位置的元素替换为新的元素,并返回被替换的旧元素。

get方法

public E get(int index) {
    rangeCheck(index);

    return elementData(index);
}

  ArrayList类提供了get方法用于获取指定位置的元素。该方法首先需要检查指定位置是否越界,然后返回指定位置的元素。

size方法

public int size() {
    return size;
}

  ArrayList类提供了size方法用于获取ArrayList中元素的数量,即size属性的值。

测试用例

  根据如上对ArrayList集合的理论知识进行了讲解之后,如下我们将通过写一个实测来辅助大家进行理解,到底如何使用Java中的ArrayList类。

测试代码演示

package com.example.javase.collection;

import java.util.ArrayList;
import java.util.List;

/**
 * @Author ms
 * @Date 2023-10-24 19:31
 */
public class ArrayListTest {
    public static void main(String[] args) {
        // create an empty arraylist
        List<String> list = new ArrayList<>();

        // add elements to the list
        list.add("Java");
        list.add("Python");
        list.add("C++");

        // print the list
        System.out.println("Original list: " + list);

        // insert an element at specific index
        list.add(1, "JavaScript");
        System.out.println("After insert an element: " + list);

        // get an element at specific index
        String element = list.get(2);
        System.out.println("Element at index 2: " + element);

        // remove an element at specific index
        list.remove(1);
        System.out.println("After remove an element: " + list);

        // remove an element by value
        list.remove("Java");
        System.out.println("After remove an element by value: " + list);
    }
}

  上述测试用例演示了ArrayList的基本用法,包括添加元素、插入元素、获取元素、删除元素等等。读者可以根据需要进行修改和拓展,以更好地理解和使用ArrayList。

测试代码结果

根据如上测试用例,测试结果如下:仅供参考:

在这里插入图片描述

测试代码分析

  根据我们如上写的测试用例代码,我们在此进行一波详细解析,大家请看:

  如上测试用例代码演示了如何使用Java中的ArrayList类。首先,创建了一个空的ArrayList对象,并通过add方法添加了三个字符串元素。然后,演示如何在指定的索引位置插入元素,获取特定索引位置的元素,以及如何删除指定索引的元素和特定值的元素。最后,通过输出语句展示了ArrayList列表的变化过程。

小结

  • ArrayList是Java集合框架中非常常用的一个动态数组,可以存储任意类型的对象。
  • ArrayList支持动态扩容和随机访问,因此在需要动态添加或删除元素,并需要支持快速随机访问的场景下,是一个非常方便的工具。
  • ArrayList的源代码非常长,但其实现原理比较简单,主要涉及到数组的扩容和元素的移动等操作。
  • ArrayList具有很高的操作效率,但在删除和插入元素时,需要进行数组元素的移动和复制,因此速度较慢。
  • ArrayList是一个线程不安全的类,因此在多线程环境下需要采取额外的措施保证线程安全。

总结

  ArrayList作为Java集合框架中的一个重要部分,在Java编程中扮演着非常重要的角色。它具有动态扩容、随机访问和丰富的 API 等特性,为开发人员提供了非常方便的数据存储和管理方式。但是,在使用ArrayList时需要注意多线程安全和操作效率的问题,在特定场景下需要选择合适的数据结构进行使用。掌握ArrayList的相关知识,对于Java开发人员来说是非常有必要的。

… …

文末

好啦,以上就是我这期的全部内容,如果有任何疑问,欢迎下方留言哦,咱们下期见。

… …

学习不分先后,知识不分多少;事无巨细,当以虚心求教;三人行,必有我师焉!!!

wished for you successed !!!


⭐️若喜欢我,就请关注我叭。

⭐️若对您有用,就请点赞叭。

⭐️若有疑问,就请评论留言告诉我叭。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【vue+el-upload】当action=“#“,代表不使用默认上传,使用自定义上传,http-request获取文件流

el-upload有多种上传行为&#xff1a; 1、立即上传&#xff1a; 当 action 属性被赋予一个有效的 URL 时&#xff0c;一旦用户选择了文件&#xff0c;el-upload 组件会立即自动将文件上传到指定的服务器地址。 2、不立即上传&#xff08;自定义触发&#xff09;&#xff1a; 如…

杰发科技AC7840——软件Sent_HAL39X

0. 序 截止2024.5.8&#xff0c;杰发的MCU没有硬件Sent功能&#xff0c;因此使用PWM模拟Sent来试试。 测试下7840的软件sent功能。 参考链接&#xff1a;SENT协议应用笔记 - TechPlus汽车工坊的文章 - 知乎 SENT协议 1. Sent功能测试 使用提供的软件Sent代码在7840上测试&a…

正点原子Linux学习笔记(五)FrameBuffer 应用编程

FrameBuffer 应用编程 19.1 什么是 FrameBuffer19.2 LCD 的基础知识19.3 LCD 应用编程介绍使用 ioctl()获取屏幕参数信息使用 mmap()将显示缓冲区映射到用户空间 19.4 LCD 应用编程练习之 LCD 基本操作19.5 LCD 应用编程练习之显示 BMP 图片在 LCD 上显示 BMP 图像在开发板上测…

超强动画制作软件blender

blender中文手册&#xff1a;Blender 4.1 Manual Blender 是一款集3D建模、渲染、动画、视频编辑、音频处理、游戏设计等多功能于一体的软件。由于其开源性质&#xff0c;它拥有庞大的用户群体和活跃的开发者社区&#xff0c;这使得Blender的功能和性能得到了不断的提升和优化…

Windows内核开发:如何使用STL

前言 大家都知道应用层c的STL非常强大&#xff0c;非常好用&#xff0c;但是在内核下就没法用了。针对这个问题&#xff0c;经过我不懈的寻找&#xff0c;终于找到了解决内核无法使用STL的方法。 使用new/delete关键字 先说一下常用关键字如何在内核中使用。其实只需要在一个全…

第四十节实现主人公的技能释放功能(二)实现技能按钮

看看我们今天要实现的效果是&#xff0c;当我们按下数字1快捷键&#xff0c;我们的技能按钮会进入倒计时&#xff0c;如下图演示&#xff1a; 一、新建场景和根节点设置 新建场景&#xff0c;选择TextureButton作为根节点&#xff0c;重名为SpellButton&#xff0c;保存场景…

啸叫抑制器采用什么处理芯片?ES56031或PH56031

会议系统或卡拉OK最头疼的就是啸叫了吧&#xff0c;来看看啸叫抑制器采用什么芯片 四通道啸叫抑制器&#xff0c;采用了2个电路板&#xff0c;每个板子处理2路信号&#xff0c;每块电路板有2个卡侬输入插座&#xff0c;2个卡侬输出插座 ES56031S&#xff0c;该啸叫抑制器为4通道…

【优选算法】——双指针——Leetcode——283.移动零

目录 ​编辑 1.题目 2. 解法&#xff08;快排的思想&#xff1a;数组划分区间-数组分两块&#xff09;&#xff1a; 1.算法思路&#xff1a; 2.算法流程&#xff1a; 3.代码实现 1.C语言 2.C 1.题目 283. 移动零 提示 给定一个数组 nums&#xff0c;编写一个函数将所有…

MySQL增删查改(进阶)

目录 数据库约束 表的设计 查询操作的进阶 查询搭配插入使用 聚合查询 1>count(*) 2>sum(*) 3>avg(*) 4>max(*) 5>min(*) group by分组分别进行聚合查询 联合查询 / 多表查询[重点] 外连接 自连接 子查询 合并查询 小结: 数据库约束 有时候…

cesium雷达扫描(消逝圆效果)

cesium雷达扫描(消逝圆效果) 以下为源码直接复制可用 1、实现思路 通过修改“material”材质来实现轨迹球效果 2、示例代码 1、index.html <!DOCTYPE html> <html lang="en"><head><!

小猪APP分发:重塑应用分发市场的创新力量

在移动互联网蓬勃发展的今天&#xff0c;应用分发平台作为连接开发者与用户的桥梁&#xff0c;扮演着至关重要的角色。然而&#xff0c;随着市场的饱和&#xff0c;如何在众多平台中脱颖而出&#xff0c;为开发者提供更宽广的舞台&#xff0c;同时确保用户能够便捷、安全地获取…

【linux】dmesg工具

dmesg介绍 dmesg工具用途&#xff1a; dmesg - print or control the kernel ring buffer kernel ring buffer, 内核环形缓冲区&#xff0c;也叫环形队列&#xff0c;Linux内核日志就存储在一个环形队列中&#xff0c;环形队列满的时候&#xff0c;新的消息会覆盖掉旧的消息。…

小程序支付的款项流转与到账时间

商家做小程序&#xff0c;最关心的是客户通过小程序下单支付的钱&#xff0c;是怎么样的流转状态以及最终到哪里。因此&#xff0c;本文将详细解析款项最终流向何处以及多久能够到账。 一、小程序支付的款项流向 当用户在小程序内完成支付后&#xff0c;款项并不会直接到达商…

CSRF漏洞简介

csrf简介 CSRF 全称为跨站请求伪造&#xff08; Cross-site request forgery &#xff09;&#xff0c;是一种网络攻击方式&#xff0c;在 CSRF 的攻击场景中攻击者会伪造一个请求&#xff08;这个请求一般是一个链接&#xff09;&#xff0c;然后欺骗目标用户进行点击&#xf…

C51版本Keil + STC-ISP 实现第一盏灯,从创建到实现

创建项目 1. 新建项目 Project -> New uVision Project 2.1 新建文件夹 2.2 输入文件名称, 并保存 3.1 选择当前位STC芯片的开发板&#xff0c;选择STC MCU Database 搜素具体芯片型号&#xff0c;进行配置&#xff1a; 3.2 选择通过搜索框搜索到stc相关芯片信息 如果st…

linux数据备份与恢复

目录 前言 1、数据备份和恢复中的两个关键性指标 2、linux系统的定时任务 1&#xff09;本地定时任务crontab 在实验测试过程中&#xff0c;遇到多次crontab任务不执行问题 &#xff0c;总结下来主要有几个方面原因&#xff1a; 2)分布式定时任务系统Jenkins 3、备份存储…

机房——蓝桥杯十三届2022国赛大学B组真题

问题分析 这题用深搜广搜都能做&#xff0c;不过我更倾向于用广搜&#xff0c;因为广搜能更容易找到目标点。那么是采用结构体存储边还是采用二维数组存储临接矩阵呢&#xff1f;我们注意到n的取值范围为1e5,用二维数组哪怕是bool类型就需要至少1e10Byte的连续空间,这个空间太大…

为软件教学文档增加实践能力

为了更方便软件教学&#xff0c;我们在凌鲨(OpenLinkSaas)上增加了公共资源引用的功能。 目前可以被引用的公共资源: 微应用常用软件公共知识库Docker模板 引用公共资源 引用微应用 目前微应用包含了主流数据库&#xff0c;终端等工具&#xff0c;可以方便的进行各种相关实…

【25届秋招备战C++】23种设计模式

【25届秋招备战C】23种设计模式 一、简介程序员的两种思维8大设计原则 二、具体23种设计模式2.1 创建型模式2.2 结构性模式2.3 行为型模式 三、常考模式的实现四、参考 一、简介 从面向对象谈起&#xff0c; 程序员的两种思维 底层思维:向下 封装&#xff1a;隐藏内部实现 多…

ASP.NET小型证券术语解释及翻译系统的设计与开发

摘 要 在系统设计上&#xff0c;综合各种翻译类型网站优缺点&#xff0c;设计出具有任何使用者都可添加术语信息的且只有管理员能够实现术语修改及删除等独特方式的术语查看管理系统。此方式能够使术语量快速增大&#xff0c;并且便于使用者及管理员操作&#xff0c;满足相互…
最新文章