深入源码解析ArrayList:探秘Java动态数组的机制与性能(1)

  1. 大小动态性:ArrayList的大小是动态的,可以根据需要动态扩展和收缩。而普通数组的大小是固定的,一旦创建就无法改变。
  2. 自动扩展:当ArrayList中的元素数量超过当前容量时,ArrayList会自动进行扩展,而普通数组需要手动重新分配内存并复制数据。
  3. 插入和删除元素效率高:ArrayList支持在任意位置插入和删除元素,而普通数组在插入和删除元素时需要移动其他元素。
  4. 内置方法和功能:ArrayList提供了许多便捷的方法和功能,如添加、删除、查找等操作,使其更易于使用和操作。

总的来说,ArrayList相对于普通数组来说更加灵活、便捷,并且具有更高的操作效率。因此,在大多数情况下,使用ArrayList比使用普通数组更加方便和实用。

二、 内部实现

2.1 数据结构:动态数组

在Java中,ArrayList是一个动态数组实现的类,它是基于数组实现的动态数组,可以自动扩容。下面是ArrayList的动态数组原理:

  1. 内部数组:ArrayList内部使用一个数组来存储元素。当创建一个ArrayList时,会初始化一个初始容量的数组。
  2. 自动扩容:当向ArrayList中添加元素时,如果当前数组已满,ArrayList会创建一个新的更大容量的数组,并将原数组中的元素复制到新数组中,然后将新元素添加到新数组中。
  3. 扩容策略:ArrayList的扩容策略是在每次扩容时将当前容量扩大为原来的1.5倍,这种策略既能够保证空间利用率,又能够减少因频繁扩容而带来的性能开销。
  4. 随机访问:由于ArrayList内部基于数组实现,因此支持随机访问,可以通过索引直接访问数组中的元素,时间复杂度为O(1)

总的来说,ArrayList通过动态扩容的方式,利用数组实现了一个动态数组,提供了高效的随机访问和动态增删元素的功能。

2.2 添加元素:add()方法的实现原理

在Java中,ArrayList的add()方法用于向ArrayList中添加元素。

其实现原理如下:

  1. 在调用add()方法时,先检查当前ArrayList的大小和容量(即存储空间是否足够)。
  2. 如果当前容量不够,就进行扩容操作。一般情况下,会创建一个新的数组,将原数组中的元素复制到新数组中,并且为新数组分配更大的存储空间。
  3. 然后将要添加的元素放入ArrayList的内部数组中,并更新ArrayList的大小。
  4. 如果添加成功,则返回true,如果添加失败,则返回false

总的来说,ArrayList的add()方法实现原理就是对内部数组的扩容和元素的添加操作。

2.3 扩容机制:ensureCapacity()方法的实现原理

在使用ArrayList时,如果我们预先知道将要插入的元素数量,可以使用ensureCapacity()方法来预先分配内部数组大小。调用了一个名为ensureCapacityInternal()的私有方法。这个方法首先会判断当前内部数组是否已经足够大来容纳新增的元素,如果不够大,则会进行扩容:

  1. 如果当前内部数组为空,则直接扩容到指定容量大小。
  2. 否则,计算出新数组的容量大小,这个容量大小取决于原始数组的大小和扩容因子(默认为1.5倍)。
  3. 如果新数组容量大小小于所需容量,则按照所需容量分配新的数组;否则,按照新数组容量大小分配新的数组。
  4. 将原始数组中的元素复制到新数组中,并将新数组赋值给ArrayList对象的elementData变量。
import java.util.ArrayList;

public class EnsureCapacityExample {
    public static void main(String[] args) {
        // 创建一个空的ArrayList
        ArrayList<String> list = new ArrayList<>();

        // 预先设定ArrayList内部数组的容量为20
        list.ensureCapacity(20);

        // 现在,ArrayList的内部数组至少可以容纳20个元素

        // 添加元素到ArrayList
        list.add("Element 1");
        list.add("Element 2");
        list.add("Element 3");

        // ...
    }
}

通过使用ensureCapacity()方法,我们可以避免由于频繁扩容带来的性能损失,提高程序效率。

三、 常见操作分析

3.1 获取元素:get()方法的实现原理

在Java中,ArrayList的get()方法实际上是通过调用数组的索引来获取指定位置的元素。ArrayList内部维护了一个Object类型的数组来存储元素。

  • 当调用get()方法时,ArrayList会将传入的索引作为数组的下标,直接访问数组中对应位置的元素,并返回该元素。
  • 因为数组的访问是基于内存地址的,所以获取元素的时间复杂度为O(1),即常数时间复杂度。
  • ArrayList的get()方法并不会对数组进行拷贝或重新分配空间,因此在获取元素时是非常高效的。
  • 频繁进行插入、删除等涉及数组扩容的操作,可能会导致性能下降。

ArrayList的get()方法通过直接访问底层数组的方式快速获取指定位置的元素。

3.2 删除元素:remove()方法的实现原理

Java中的ArrayList类是基于数组实现的动态数组,当我们使用remove()方法从ArrayList中删除元素时,这个方法会将指定位置的元素从内部数组中移除,并将后续元素向前移动一位。

这个操作可以通过以下几个步骤来实现:

  1. 检查待删除的元素下标是否越界,如果越界则抛出IndexOutOfBoundsException异常。
  2. 将要删除的元素从内部数组中移除,这个过程可以通过System.arraycopy()方法来实现,该方法可以将数组的某一范围内的元素复制到另一个位置上。
  3. 将后续元素向前移动一位,以填补被删除的空位。这个过程同样可以通过System.arraycopy()方法来实现。

ps:ArrayList的remove()方法只能移除第一个与指定元素相等的元素。如果我们想要移除所有等于指定元素的元素,可以通过循环遍历ArrayList并使用remove()方法来实现。

3.3 修改元素:set()方法的实现原理

Java中的ArrayList是一种基于数组的动态数组实现,它继承了AbstractList类并实现了List接口。set()方法是ArrayList中的一个方法,用于将指定索引位置的元素替换为新的元素。

其实现原理如下

  1. 首先,set()方法会检查传递的索引是否在ArrayList范围之内。如果索引小于0或大于等于ArrayList的大小(size()方法返回的值),则会抛出IndexOutOfBoundsException异常。
  2. 如果索引有效,则会使用数组的索引定位到指定的元素,并将其替换为新的元素。
  3. set()方法返回被替换掉的元素。
  4. 在替换元素时,ArrayList可能需要调整内部数组的大小。如果新元素的大小与当前数组的容量不匹配,ArrayList会创建一个新数组,并将所有元素从旧数组复制到新数组中。

总之,ArrayList的set()方法的实现原理是通过数组索引定位和替换元素来完成的,而且可能需要动态调整内部数组的大小。

四、 性能分析

4.1 时间复杂度分析

在Java中,ArrayList是一个动态数组实现的集合类,它提供了随机访问和快速插入/删除元素的功能。

下面是ArrayList的常见操作及其时间复杂度分析

  1. 访问元素(get):通过索引访问特定位置的元素,时间复杂度为O(1)
  2. 插入元素(add):在指定位置插入元素,平均时间复杂度为O(n),最坏情况下需要将插入位置之后的元素都向后移动,时间复杂度为O(n)
  3. 删除元素(remove):删除指定位置的元素,平均时间复杂度为O(n),最坏情况下需要将删除位置之后的元素都向前移动,时间复杂度为O(n)
  4. 查找元素(contains):判断集合中是否包含某个元素,平均时间复杂度为O(n),需要遍历整个集合来查找。
  5. 获取集合大小(size):获取集合中元素的数量,时间复杂度为O(1)

需要注意的是,ArrayList的插入和删除操作涉及到元素的移动,当集合的大小较大时,这些操作可能会导致性能下降。如果需要频繁进行插入和删除操作,可以考虑使用LinkedList来代替ArrayList,因为LinkedList对于插入和删除操作的时间复杂度是O(1)

4.2 空间复杂度分析

ArrayList的空间复杂度主要取决于两个因素:集合中的元素数量和内部数组的容量。

  1. 元素数量:ArrayList存储的元素数量,即集合的大小,会占用一定的空间。假设元素数量为n,则空间复杂度为O(n)
  2. 内部数组容量:ArrayList内部使用一个动态数组来存储元素,数组的容量可能会比集合的大小大一些,以容纳未来添加的元素。假设数组的容量为m,则空间复杂度为O(m)

需要注意的是,ArrayList的实际空间占用可能会比集合中的元素数量多一些,因为它预留了一些额外的容量供后续添加元素使用。当集合的元素数量接近或超过内部数组的容量时,ArrayList会自动进行扩容操作,重新分配更大的数组并将原有元素复制到新数组中,这可能会导致空间复杂度的增加。

在实际使用中,可以通过调整ArrayList的初始容量或使用构造函数指定初始容量来控制空间复杂度。通常情况下,如果能够预估集合的大小,设置一个适当的初始容量可以减少扩容操作的频率,提高性能。

4.3 与LinkedList的比较

ArrayList和LinkedList是Java中两种常见的集合类,它们都实现了List接口,但在内部实现和性能特点上有所不同。

下面是ArrayList和LinkedList的比较

  1. 内部实现
    • ArrayList:使用动态数组实现,内部维护一个可变长度的数组来存储元素。
    • LinkedList:使用双向链表实现,内部由一系列节点组成,每个节点包含元素值和前后指针。
  2. 访问效率
    • ArrayList:由于使用数组实现,可以通过索引直接访问元素,因此随机访问的效率很高,时间复杂度为O(1)。但在插入和删除元素时,需要移动数组中的元素,效率较低,时间复杂度为O(n)
    • LinkedList:插入和删除元素的效率较高,因为只需要调整节点的指针,时间复杂度为O(1)。但在随机访问元素时,需要从头节点开始按序遍历查找,效率较低,时间复杂度为O(n)
  3. 空间占用
    • ArrayList:使用动态数组,会预留一定容量的空间,当元素数量超过容量时,需要进行扩容操作。因此,可能会有额外的空间浪费。
    • LinkedList:使用链表结构,每个节点除了存储元素还需要存储前后节点的指针,因此会略微增加一些额外空间。
  4. 适用场景:
    • ArrayList:适合于随机访问和遍历操作较多的场景,例如根据索引访问元素、遍历集合等。但在频繁插入和删除元素的情况下,性能相对较差。
    • LinkedList:适合于频繁插入和删除元素的场景,例如实现队列或栈等数据结构。但在随机访问元素时,性能相对较差。

五、 源码解读

5.1 成员变量

在这里插入图片描述

5.2 构造方法

在这里插入图片描述

5.3 trimToSize()方法

在这里插入图片描述

5.4 indexOf()方法

在这里插入图片描述

5.5 clone()方法

在这里插入图片描述

5.6 get()方法

在这里插入图片描述

5.7 set()方法

在这里插入图片描述

5.8 add()方法

在这里插入图片描述

5.9 remove()方法

在这里插入图片描述

5.10 addAll()方法

在这里插入图片描述

六、 案例分析与实例演示

6.1 案例分析

假设有一个学生管理系统,需要存储学生的信息,包括姓名、年龄、性别等。

为了方便管理,我们可以使用ArrayList来存储学生对象。

首先定义一个学生类,包含姓名、年龄、性别三个属性

public class Student {
    private String name;
    private int age;
    private String gender;

    public Student(String name, int age, String gender) {
        this.name = name;
        this.age = age;
        this.gender = gender;
    }

    // getter 和 setter 方法省略
}

然后在主类中创建ArrayList对象,并添加学生信息

import java.util.ArrayList;

public class Main {
    public static void main(String[] args) {
        // 创建ArrayList对象
        ArrayList<Student> list = new ArrayList<>();

        // 添加学生信息
        list.add(new Student("张三", 18, "男"));
        list.add(new Student("李四", 20, "女"));
        list.add(new Student("王五", 19, "男"));

        // 遍历学生信息
        for (Student student : list) {
            System.out.println("姓名:" + student.getName() + " 年龄:" + student.getAge() + " 性别:" + student.getGender());
        }
    }
}

输出结果如下

姓名:张三 年龄:18 性别:男
姓名:李四 年龄:20 性别:女
 **自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。**

**深知大多数Python工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!**

**因此收集整理了一份《2024年Python开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。**

![img](https://img-blog.csdnimg.cn/img_convert/005fabca40094890d2db3ef027e29100.png)

![img](https://img-blog.csdnimg.cn/img_convert/d49336e8f7ca33b67b500bf43c3985f9.png)

![img](https://img-blog.csdnimg.cn/img_convert/350cae6cfcf1e947c883edaa20154594.png)

![img](https://img-blog.csdnimg.cn/img_convert/a04ddf9fd17218ad7234287be0db2519.png)

![img](https://img-blog.csdnimg.cn/img_convert/6c361282296f86381401c05e862fe4e9.png)

![img](https://img-blog.csdnimg.cn/img_convert/9f49b566129f47b8a67243c1008edf79.png)

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!**

**由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新**

**如果你觉得这些内容对你有帮助,可以扫码获取!!!(备注:Python)**

tps://img-blog.csdnimg.cn/img_convert/6c361282296f86381401c05e862fe4e9.png)

![img](https://img-blog.csdnimg.cn/img_convert/9f49b566129f47b8a67243c1008edf79.png)

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!**

**由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新**

**如果你觉得这些内容对你有帮助,可以扫码获取!!!(备注:Python)**

![](https://img-blog.csdnimg.cn/img_convert/c3e97c2f9633d141e8fb5d32b3d817eb.jpeg)
  • 18
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值