Java ArrayList的底层实现方法

Java中的ArrayList是一种动态数组数据结构,底层通过数组实现,其大小可以随时增加或缩小。ArrayList可以存储任何类型的数据,而不仅仅是对象。下面将介绍Java ArrayList的底层实现方法。

一、数据结构

ArrayList底层的数据结构是数组,其构造方法为:

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

其中elementData是一个Object类型的数组,表示ArrayList中存储元素的数组,而DEFAULTCAPACITY_EMPTY_ELEMENTDATA是一个空数组,初始大小为0.

在添加元素时,ArrayList首先检查elementData数组是否为空,如果是,则重新创建一个大小为10的数组。

private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
private static final int DEFAULT_CAPACITY = 10;
public boolean add(E e) {
    ensureCapacityInternal(size + 1);
    elementData[size++] = e;
    return true;
}
private void ensureCapacityInternal(int minCapacity) {
        ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}
private void ensureExplicitCapacity(int minCapacity) {
        modCount++;
        // overflow-conscious code
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
}
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);
    elementData = Arrays.copyOf(elementData, newCapacity);
}

可以发现,当添加元素时,ArrayList会调用ensureCapacityInternal方法来确保elementData数组的容量足够,如果不够则调用grow方法进行扩容。grow方法会创建一个新的数组,并将旧数组中的元素复制到新数组中。

二、例子

添加元素

ArrayList<String> list = new ArrayList<>();
list.add("Hello");
list.add("World");

在这个例子中,我们创建了一个ArrayList对象,并向其中添加了两个元素。当添加第一个元素时,elementData数组为空,ArrayList会创建一个大小为10的数组。当添加第二个元素时,elementData数组大小为1,由于ArrayList底层实现为动态数组,因此会自动扩容为20,将旧数组中的元素复制到新数组中,并添加新的元素。

删除元素

list.remove(0);

在这个例子中,我们通过索引删除了ArrayList中的第一个元素。ArrayList删除元素会涉及到数组的移位操作,因此效率不如添加元素高。

三、总结

Java ArrayList底层实现方法是通过动态数组来实现的,其数据结构为数组,每次添加或删除元素时,ArrayList会检查elementData数组容量是否足够,如果不够则扩容并复制旧数组中的元素到新数组中。由于ArrayList底层实现为数组,删除元素需要涉及到数组的移位操作,因此效率不高。

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:Java ArrayList的底层实现方法 - Python技术站

(0)
上一篇 2023年5月26日
下一篇 2023年5月26日

相关文章

  • Spring MVC请求参数接收的全面总结教程

    接下来我将详细讲解Spring MVC请求参数接收的全面总结教程。 为什么需要请求参数接收 在Web开发中,经常需要接收前端传来的数据,这些数据以请求参数的形式传递。请求参数通常包含了用户请求的具体行为,并提供了必要的参数数据。例如,访问百度搜索,连接中会携带请求参数q,表示搜索关键词。 Spring MVC框架提供了有用且全面的请求参数接收处理机制,让我们…

    Java 2023年5月16日
    00
  • Java Spring框架简介与Spring IOC详解

    Java Spring框架简介 Spring是一个轻量级的Java框架,用于构建企业级应用程序。它提供了一系列的模块,包括Spring Core、Spring MVC、Spring Data、Spring Security等,可以帮助开发人员快速构建高质量的应用程序。 Spring框架的主要特点包括: 轻量级:Spring框架本身非常轻量级,不需要依赖其他的…

    Java 2023年5月18日
    00
  • Spring 数据库连接池(JDBC)详解

    Spring 数据库连接池(JDBC)详解 什么是数据库连接池 在开发 Web 应用程序时,通常会使用数据库进行数据存储和管理。当客户端通过应用程序访问数据库时,应用程序需要使用 JDBC 连接到数据库并执行查询或更新操作。在实际开发中,频繁地创建和关闭连接非常消耗资源并且会影响应用程序的性能。为此,使用连接池可以提高性能并减少资源消耗。 数据库连接池是管理…

    Java 2023年5月20日
    00
  • IDEA上运行Flink任务的实战教程

    下面是“IDEA上运行Flink任务的实战教程”的完整攻略: 1. 环境要求 在开始之前,我们需要先完成以下环境的搭建: Java环境。需要安装Java 8以上版本。 IDEA。需要安装适用于Java开发的IDEA软件,版本要求为2019.3及以上版本。 Flink。需要下载安装Flink,版本要求为1.11及以上版本。 2. 创建Flink项目 在IDEA…

    Java 2023年5月20日
    00
  • JQuery弹出层示例可自定义

    现在我来给您详细讲解如何实现一个可自定义的jQuery弹出层示例。 1. 准备工作 在使用jQuery之前,我们需要先引入jQuery库文件。一般情况下,我们可以下载jQuery库到本地,然后在要使用的网页中引入。例如: <script src="jquery.min.js"></script> 2. 自定义弹出层…

    Java 2023年6月15日
    00
  • java实现猜拳游戏试题

    下面我将详细讲解“java实现猜拳游戏试题”的完整攻略。 1. 确定游戏规则 在开始编写程序之前,需要先确定猜拳游戏的规则。通常猜拳游戏有剪刀、石头和布三种手势,其中剪刀克制布,布克制石头,石头克制剪刀。参与游戏的两个玩家选择其中一种手势,如果两个玩家选择的手势相同,则为平局;否则根据手势的胜负关系判断胜负,并输出胜负结果。 2. 编写程序 2.1. 实现游…

    Java 2023年5月23日
    00
  • java获取properties属性文件示例

    当我们需要在Java程序中读取properties属性文件时,通常可以使用java.util.Properties类来实现。下面是实现此操作的完整攻略: 1. 获取properties文件 首先需要获取到带有相关属性的properties文件,可以通过在项目中创建文件或者从外部导入文件的方式进行获取。假设我们已经有了一个示例属性文件”example.prop…

    Java 2023年5月19日
    00
  • Java利用递归算法实现查询斐波那契数

    下面我将详细讲解Java利用递归算法实现查询斐波那契数的完整攻略。 什么是斐波那契数 斐波那契数指的是一个数列,该数列从第3项开始每一项都等于前两项之和。这个数列如下所示:0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, …,通常用F(n)表示该数列的第n项。 利用递归算法实现查询斐波那契数 递归是一种通过自身调用来实现循…

    Java 2023年5月19日
    00
合作推广
合作推广
分享本页
返回顶部