ArrayList详解

news2025/1/11 21:01:07

ArrayList是什么?

ArrayList就是动态数组,是List接口的可调整大小的数组实现;除了实现List接口之外,该类还提供了一些方法来操纵内部使用的存储列表的数组大小。它的主要底层实现是数组Object[] elementData。

为什么要设计ArrayList?

大家在使用数组的时候需要在初始化的时候去指定这个数组的长度,对于元素数量不确定的场景如果使用数组则需要预估所需的数组容量,如果估少了还需要重新申请空间并对原数组进行拷贝,因此ArrayList应运而生,用来解决数组无法动态扩容的缺陷,实现了动态扩容的效果。

数组的特点

遍历查询速度快——数组在内存是连续空间,可以根据地址+索引的方式快速获取对应位置上的元素。但是它的增删速度慢——每次删除元素,都需要更改数组长度、拷贝以及移动元素位置。

ArrayList的特性

1.允许插入的元素重复
2.插入的元素是有序的
3.动态扩容
4.非线程安全,异步
5.基于动态数组的数据结构
6.擅长随机访问(get set)

类结构图

在这里插入图片描述
ArrayList 是 java 集合框架中比较常用的数据结构了。继承自 AbstractList ,实现了 List 接口。底层基于数组实现容量大小动态变化。允许 null 的存在。同时还实现了 RandomAccess、Cloneable、Serializable 接口,所以ArrayList 是支持快速访问、复制、序列化的。

常用方法

add(Object e)
add(int index ,Object e)
addAll(Collection c)
addAll(int index , Collection c)
size()
get(int index)
set(int index,Object e)
indexOf(Object c)
lastIndexOf(Object c)
isEmpty()
remove(int index)
remove(Object c)
removeAll(Collection<?> c) contains(Object c) containsAll(Collection<?> c)
clear()
clone()
iterator()
retainAll(Collection<?> c)
subList(int fromIndex,int toIndex)
trimToSize() 回收多余容量
toArray()
toArray(T[] a)

部分方法使用案例

 
    @Test
    public void addTest() {
        ArrayList list = new ArrayList();
        list.add(0);
        list.add(1);
        list.add(2);
        list.add(3); // 此时list : 0 1 2 3
        print("list", list);
        list.add(0, 4); //  在指定位置插入元素
        print("list", list); //  此时list ;4 0 1 2 3
        List newList = list.subList(2, 4); //截取list ,不包含最后一位
        print("newList", newList); //  此时 newList : 1 2
        list.addAll(2, newList); //新增list,新的list元素默认排在原来元素的后面 ,此时newList 不能再使用了
        print("list", list); // 此时,list : 4 0 1 2 1 2 3
        System.out.println("list size " + list.size());
        list.set(0, 1); //在指定位置替换元素
        print("list", list); // 此时 list : 1 0 1 2 1 2 3
        System.out.println(list.indexOf(1));
        System.out.println(list.lastIndexOf(1));
        System.out.println(list.isEmpty());
 
    }
 
    private void print(String flag, List list) {
        System.out.print(flag + ":");
        list.forEach(System.out::print);
        System.out.println();
    }

构造方法

无参构造

示例

  private List<User> list = new ArrayList<>();  
  

源码

    //初始化一个空数组
    public ArrayList() {
        this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
    }
    
    private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};

ArrayList(int initialCapacity)

构造具有指定初始容量的List

示例

    private List<User> list = new ArrayList<>(10);

源码

    public ArrayList(int initialCapacity) {
        if (initialCapacity > 0) {
            //将传入的 10作为参数初始化数组长度
            this.elementData = new Object[initialCapacity];
        } else if (initialCapacity == 0) {
            this.elementData = EMPTY_ELEMENTDATA;
        } else {
            throw new IllegalArgumentException("Illegal Capacity: "+
                                               initialCapacity);
        }
    }

ArrayList(Collection<? extends E> c)

构造一个包含指定集合的元素的列表,按照它们由集合迭代器返回的顺序

案例

    private List<User> list = new ArrayList<>(new ArrayList<>());

源码

    public ArrayList(Collection<? extends E> c) {
        // 将传入的List 赋值给 elementData 
        elementData = c.toArray();
        //如果size != 0 代表有元素
        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 {
           // size== 0 初始化空数组 
            // replace with empty array.
            this.elementData = EMPTY_ELEMENTDATA;
        }
    }

add()方法解析

案例

private List<String> list = new ArrayList<>();
  list.add("test");

源码

添加元素e到容器

    public boolean add(E e) {
       //确保对象数组elementData有足够的容量,可以将新加入的元素e加进去
        ensureCapacityInternal(size + 1);  // Increments modCount!!
        //加入新元素e,size加1
        elementData[size++] = e;
        return true;
    }
       

调用方法对内部容量进行校验

      private void ensureCapacityInternal(int minCapacity) {
       //判断集合存数据的数组是否等于空容量的数组 
        if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
         //通过最小容量和默认容量 出较大值 (用于第一次扩容)
            minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
        }
         //将if中计算出来的容量传递给下一个方法,继续校验 
        ensureExplicitCapacity(minCapacity);
    }

确保数组的容量足够存放新加入的元素,若不够,要扩容

    private void ensureExplicitCapacity(int minCapacity) {
    //实际修改集合次数++ (在扩容的过程中没用,主要是用于迭代器中)
        modCount++;

        // overflow-conscious code
         //判断最小容量    - 数组长度是否大于    0 
        if (minCapacity - elementData.length > 0)
           //扩容
            grow(minCapacity);
    }

扩容

    private void grow(int minCapacity) {
        // overflow-conscious code
        //先获取数组的长度
        int oldCapacity = elementData.length;
        //在原有长度的基础上扩容1.5倍
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        //判断新容量-最小容量是否小于0, 如果是第一次调用add方法必然小于
        if (newCapacity - minCapacity < 0)
            //还是将最小容量赋值给新容量
            newCapacity = minCapacity;
           //判断新容量-最大数组大小是否>0,如果条件满足就计算出一个超大容
        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);
    }

获取最大值

private static int hugeCapacity(int minCapacity) {
       // 如果数组个数小于0抛出OutOfMemoryError异常
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();       
       // 如果最数组个数大于最大容量 就返回最大值,否则返回最大容量
        return (minCapacity > MAX_ARRAY_SIZE) ?
            Integer.MAX_VALUE :
            MAX_ARRAY_SIZE;
}

get(int index)

通过下标获取元素

案例

private List<String> list = new ArrayList<>();
 for (int i = 0; i <list.size() ; i++) {
                String s = list.get(i);
            }

源码

    public E get(int index) {
        rangeCheck(index);
          
        return elementData(index);
    }

校验该下标是否超过数组长度

    private void rangeCheck(int index) {
        //如果下标超过数组长度直接报数组越界异常
        if (index >= size)
            throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
    }

通过下标获取元素

    E elementData(int index) {
        //通过下表从数组中获取元素
        return (E) elementData[index];
    }

remove()删除元素

public E remove(int index) {
       //指定位置检查
        rangeCheck(index);
        modCount++;       
       //保留要删除的值
        E oldValue = elementData(index);
       //要移动元素个数
        int numMoved = size - index - 1;
        if (numMoved > 0)
        //通过拷贝使数组内位置为 index+1到 (size-1)的元素往前移动一位
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        //清除末尾元素让GC回收
        elementData[--size] = null; // clear to let GC do its work
        //返回删除的值
        return oldValue;
    }

ArrayList动态扩容

扩容的大小是原先数组的1.5倍;
若值newCapacity比传入值minCapacity还要小,则使用传入minCapacity,若newCapacity比设定的最大容量大,则使用最大整数值;

    private void grow(int minCapacity) {
        // overflow-conscious code
        //先获取数组的长度
        int oldCapacity = elementData.length;
        //在原有长度的基础上扩容1.5倍
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        //判断新容量-最小容量是否小于0, 如果是第一次调用add方法必然小于
        if (newCapacity - minCapacity < 0)
            //还是将最小容量赋值给新容量
            newCapacity = minCapacity;
           //判断新容量-最大数组大小是否>0,如果条件满足就计算出一个超大容
        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);
    }

为什么扩容是扩充1.5倍?

可以充分利用前面已经释放的空间 如果大于等于2 心容量刚好永远大于过去所有废弃的数组容量
1.5可以充分利用位移运算减少浮点数或者运算时间和运算次数

解决频繁扩容导致的性能低问题

ArrayList底层是数组实现的,那么每次添加数据时会不断地扩容,这样的话会占内存,性能低,所以导致时间很长。
我们可以用ArrayList的指定初始化容量的构造方法来解决性能低的问题。

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

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

相关文章

【C++】mapset利用红黑树进行简单封装

前言 大家好~~~~呀&#xff01;很荣幸你能点击这篇文章。本篇也是我的一份学习笔记&#xff0c;让我们一起共同成长吧~ing...... C红黑树的简单插入实现博客~ 【C】红黑树的插入实现_柒海啦的博客-CSDN博客 二叉搜索树的基本结构和实现博客~ 【C】二叉搜索树_柒海啦的博客-CSDN…

java 实现一个最小栈

文章目录最小栈1.实现思路2.实现过程演示3.代码实现思路3.1 压入思路3.2 弹出思路3.3 如何返回栈顶元素的下标3.4 如何返回栈的最小值4.整体代码实现最小栈 1.实现思路 实现一个stack栈 和 minStack栈。先将数据一个一个压入到 stack 中。找到 stack 中的最小值。minStack中始…

简单介绍动态链接过程

文章目录gotgot[0] link_map结构体地址got[1] _dl_runtime_resolvegot[2]之后pltplt[0] 调用libc解析函数plt后面的plt.sec随便拿ida打开一个程序可以看到这是got的内容gdb一下查看内容&#xff0c;可以看到地址是从0开始的大家也知道 got是个独立的section&#xff0c;所以最开…

MySQL数据库(Java的数据库编程:JDBC)

作者&#xff1a;渴望力量的土狗 博客主页&#xff1a;渴望力量的土狗的博客主页 专栏&#xff1a;MySQL数据库 目录 什么是数据库编程&#xff1a; 什么是JDBC? JDBC工作原理&#xff1a; JDBC的使用及相关操作&#xff1a; JDBC开发案例&#xff1a; JDBC常用接口…

关于电影的HTML网页设计-威海影视网站首页-电影主题HTM5网页设计作业成品

HTML实例网页代码, 本实例适合于初学HTML的同学。该实例里面有设置了css的样式设置&#xff0c;有div的样式格局&#xff0c;这个实例比较全面&#xff0c;有助于同学的学习,本文将介绍如何通过从头开始设计个人网站并将其转换为代码的过程来实践设计。 文章目录一、网页介绍一…

5分钟教你如何设计一个安全web架构

今天就给大家聊聊web安全&#xff0c;web安全占比还是比较大的&#xff0c;基础的从一些html标签&#xff0c;到js 然后到接口&#xff0c;数据库&#xff0c;以及流量攻击&#xff0c;模拟请求。当然这也谈到了一个概念&#xff0c;全新的架构设计模式&#xff0c;前后端分离&…

一文讲解如何学习 Linux 内核网络协议栈

协议栈的细节 下面将介绍一些内核网络协议栈中常常涉及到的概念。 sk_buff 内核显然需要一个数据结构来表示报文&#xff0c;这个结构就是 sk_buff ( socket buffer 的简称)&#xff0c;它等同于在<TCP/IP详解 卷2>中描述的 BSD 内核中的 mbuf。 sk_buff 结构自身并不…

【毕业设计】深度学习人脸性别年龄识别系统 - python

文章目录0 前言1 课题描述2 实现效果3 算法实现原理3.1 数据集3.2 深度学习识别算法3.3 特征提取主干网络3.4 总体实现流程4 具体实现4.1 预训练数据格式4.2 部分实现代码5 最后0 前言 &#x1f525; Hi&#xff0c;大家好&#xff0c;这里是丹成学长的毕设系列文章&#xff0…

【案例分享】华为防火墙出接口方式的单服务器智能DNS配置

介绍出接口方式的单服务器智能DNS的配置举例。 组网需求 如图1所示&#xff0c;企业部署了一台ISP1服务器对外提供Web服务&#xff0c;域名为www.example.com。ISP1服务器的私网IP地址为10.1.1.10&#xff0c;服务器映射后的公网IP地址为1.1.1.10。企业的DNS服务器上存在域名w…

为什么你的用户转化率不高?-- 新媒体运营转化效果渠道归因分析

新媒体运营人最关注的就是流量和用户转化问题。公司发布了新APP、上线了新网站项目&#xff0c;进行用户定位、策划、数据分析和内容营销&#xff0c;花重钱做产品推广&#xff0c;但最后用户转化率却不高&#xff0c;大批用户流失了......这种现象是运营人最不愿意看到的&…

老杨说运维|今年这个会议非比寻常

前言&#xff1a; 人民银行印发的《金融科技(FinTech)发展规划(2022-2025年)》中&#xff0c;重点围绕数字化转型建设&#xff0c;强调上云、数据基础建设以及数智应用的重要性&#xff0c;明确了金融科技的长期重点建设方向。 由金科创新社主办的“2022金融业新一代数据中心发…

kwebio/kweb-core:面向后端的轻量级 Kotlin Web 框架

现代网站至少由两个紧密耦合 的组件组成&#xff0c;一个在浏览器中运行&#xff0c;另一个在服务器上。它们通常用不同的编程语言编写&#xff0c;并且必须通过 HTTP(S) 连接相互通信。 Kweb 的目标是消除这种服务器/浏览器分离&#xff0c;这样您就可以专注于构建您的网站或用…

react多组件出错其他正常显示

问题&#xff1a;一个组件内部有很多个子组件&#xff0c;其中一个出错&#xff0c;怎么实现其他组件可以正常显示&#xff0c;而不是页面挂掉&#xff1f; 一、错误边界 可以捕获发生在其子组件树任何位置的 JavaScript 错误&#xff0c;并打印这些错误&#xff0c;同时展示…

CC攻击和DDOS攻击哪个对服务器影响更大

互联网企业&#xff0c;不管是小企业&#xff0c;还是大企业&#xff0c;大多数企业网站都遭受过攻击&#xff0c;而我们时不时的也能在网上看见某大型企业网站被攻击&#xff0c;崩溃的新闻&#xff0c;网络攻击可以说是屡见不鲜了。攻击力最常见的就是DDOS攻击和CC攻击&#…

使用HTML+CSS技术制作篮球明星介绍网站

&#x1f389;精彩专栏推荐 &#x1f4ad;文末获取联系 ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 &#x1f482; 作者主页: 【主页——&#x1f680;获取更多优质源码】 &#x1f393; web前端期末大作业&#xff1a; 【&#x1f4da;毕设项目精品实战案例 (10…

2022年数维杯国际数模赛浅评

今日数维杯国际大学生数学建模挑战赛将要开赛&#xff0c;为了更好的帮助大家整理了以下注意事项&#xff0c; 竞赛开始时间&#xff1a;北京时间2022年11月17日08:00&#xff08;周四&#xff09; 竞赛结束时间&#xff1a;北京时间2022年11月21日08&#xff1a;00&#xff…

ffmpeg视频编解码 demo初探(二)(包含下载指定windows版本ffmpeg)将YUV图片序列作为流读入,编码封装成x264 MP4视频

参考文章&#xff1a;【FFmpeg编码实战】&#xff08;1&#xff09;将YUV420P图片集编码成H.264视频文件 文章目录第二个项目&#xff1a;将YUV图片序列作为流读入&#xff0c;编码封装成x264 MP4视频将YUV图片序列编码成.h264文件将YUV图片序列编码成mp4文件第二个项目&#x…

艾美捷测序级 II,纯化胰蛋白酶化验程序文献参考

胰蛋白酶是一种基于带正电荷的赖氨酸和精氨酸侧链的底物特异性胰丝氨酸蛋白酶&#xff08;Brown and Wold 1973&#xff09;。这种酶由胰腺排出&#xff0c;参与食物蛋白质的消化和其他生物过程。胰蛋白酶是一种中等大小的球状蛋白&#xff0c;作为一种无活性的胰蛋白酶原产生&…

甘露糖-顺铂mannose-cisplatin|甘露糖-聚乙二醇-顺铂cisplatin-PEG-mannose

甘露糖-顺铂mannose-cisplatin|甘露糖-聚乙二醇-顺铂cisplatin-PEG-mannose 顺铂&#xff0c;又名顺式-二氯二氨合铂&#xff0c;是一种含铂的药物&#xff0c;呈橙黄色或黄色结晶性粉末&#xff0c;微溶于水、易溶于二甲基甲酰胺&#xff0c;在水溶液中可逐渐转化成反式和水解…

基于Feign接口的全链路拦截器

1、前言 单体应用时&#xff0c;我们经常会把一些共享数据&#xff0c;比如登录信息等放在session里面&#xff0c;当然也可以放在ThreadLocal里面。随着业务复杂度的提高&#xff0c;分布式应用越来越主流。单机的存储的思想已经不适用了&#xff0c;共享session应运而生&…