Java List系列(ArrayList、LinekdList 以及遍历中删除重复元素时发生的异常和解决办法)

news2025/4/11 1:24:20

目录

      • List集合系列
        • List系列集合特点
        • List集合特有方法
        • List集合的遍历方式
        • ArrayList集合的底层原理
          • 分析源码
        • LinkedList集合的底层原理
        • 集合的并发修改异常问题(删除重复元素时)

List集合系列

在这里插入图片描述

List系列集合特点

ArrayList、LinekdList :有序,可重复,有索引。

  • 有序:存储和取出的元素顺序一致
  • 有索引:可以通过索引操作元素
  • 可重复:存储的元素可以重复

List集合特有方法

List集合因为支持索引,所以多了很多索引操作的独特api,其他Collection的功能List也都继承了。
在这里插入图片描述

在这里插入图片描述

List集合的遍历方式

List集合的遍历方式有几种?

  1. 迭代器 (Collection集合)
  2. 增强for循环(Collection集合)
  3. Lambda表达式(Collection集合)
  4. for循环(因为List集合存在索引,独有)
public static void main(String[] args) {
        List<String> list=new ArrayList<>();
        list.add("String1");
        list.add("String2");
        list.add("String3");
        list.add("String4");

        //for 循环遍历
        for (int i = 0; i < list.size(); i++) {
            String element=list.get(i);
            System.out.println(element);
        }
    }

ArrayList集合的底层原理

ArrayList底层是基于数组实现的:根据索引定位元素快,增删需要做元素的移位操作。

分析源码

第一次创建集合并添加第一个元素的时候,在底层创建一个默认长度为10的数组。

    /**
     * Default initial capacity.
     */
    private static final int DEFAULT_CAPACITY = 10;

什么时候扩容:

    private void add(E e, Object[] elementData, int s) {
        if (s == elementData.length)//s为当前集合的长度,当它等于现在集合的最大长度时,调用grow()方法扩容
        elementData = grow();
        elementData[s] = e;//否则就插入集合
        size = s + 1;//集合当前长度加一
    }

如何扩容:

    public static final int SOFT_MAX_ARRAY_LENGTH = Integer.MAX_VALUE - 8;

    private Object[] grow(int minCapacity) {//minCapcity=size+1
        int oldCapacity = elementData.length;
        if (oldCapacity > 0 || elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
            int newCapacity = ArraysSupport.newLength(oldCapacity,
                    minCapacity - oldCapacity, /* minimum growth */
                    oldCapacity >> 1           /* preferred growth */);
            return elementData = Arrays.copyOf(elementData, newCapacity);
        } else {
            return elementData = new Object[Math.max(DEFAULT_CAPACITY, minCapacity)];
        }
    }

    private Object[] grow() {
        return grow(size + 1);
    }

    public static int newLength(int oldLength, int minGrowth, int prefGrowth) {
        // preconditions not checked because of inlining
        // assert oldLength >= 0
        // assert minGrowth > 0

        int prefLength = oldLength + Math.max(minGrowth, prefGrowth); // might overflow
        if (0 < prefLength && prefLength <= SOFT_MAX_ARRAY_LENGTH) {
            return prefLength;
        } else {
            // put code cold in a separate method
            return hugeLength(oldLength, minGrowth);
        }
    }

当ArrayList的容量扩容1.5倍后不会超过规定的最大值,则扩大为原来的1.5倍,即newCapacity=oldCapacity+(oldCapacity >> 1)
当ArrayList的容量扩容1.5后超过规定的最大值,则新的容量为原来的容量加上此次要增加的容量,调append()时,加的恰好为1,即newCapacity=oldCapacity+(minCapacity - oldCapacity)

综上,AraayList的扩容刚开始每次是原来的1.5倍,超过一个临界点会根据需求扩容。

    public static <T,U> T[] copyOf(U[] original, int newLength, Class<? extends T[]> newType) {
        @SuppressWarnings("unchecked")
        T[] copy = ((Object)newType == (Object)Object[].class)
            ? (T[]) new Object[newLength]
            : (T[]) Array.newInstance(newType.getComponentType(), newLength);
        System.arraycopy(original, 0, copy, 0,
                         Math.min(original.length, newLength));
        return copy;
    }
    @IntrinsicCandidate
    public static native void arraycopy(Object src,  int  srcPos,
                                        Object dest, int destPos,
                                        int length);

并且,可以看到数据最后是调用了System.arraycopy()方法,来将原来的数据复制到新的数组中。

LinkedList集合的底层原理

底层数据结构是双链表,查询慢,首尾操作的速度是极快的,所以多了很多首尾操作的特有API。

在这里插入图片描述

集合的并发修改异常问题(删除重复元素时)

当我们从集合中找出某个元素并删除的时候可能出现一种并发修改异常问题。
那么哪些遍历存在问题?

  1. 迭代器遍历集合且直接用集合删除重复元素的时候可能出现。
  2. 增强for循环遍历集合且直接用集合删除重复元素的时候可能出现。
    public static void main(String[] args) {
        ArrayList<String> list=new ArrayList<>();
        list.add("String1");
        list.add("String1");
        list.add("String2");
        list.add("String3");

        Iterator<String> i= list.iterator();
        while (i.hasNext()){
            String element=i.next();
            if ("String1".equals(element)){
                //list.remove(element);//会报异常,原因集合会自动前移,会漏删
                i.remove();//正确:使用迭代器删除,内部实现了--
            }
        }
        System.out.println(list);
    }

迭代器遍历集合但是用迭代器自己的删除方法操作可以解决。
增强for无法解决
for循环可以从后往前删或则每次删除减一

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

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

相关文章

HNU工训中心:电子开关与信号隔离

工训中心的牛马实验 1.实验目的&#xff1a; 1) 认识三极管和MOS管构成三端电子开关电路&#xff1b; 认识信号隔离的继电器和光电隔离方式。 2) 认识施密特触发器&#xff0c;掌握一种波形变换方法。 3) 实现一种脉冲波形发生器。 2.实验资源 HBE硬件基础电路实验箱、示波…

2.FFmpeg5.1下载和使用

1.FFmpeg库下载 进入http://ffmpeg.org/download.html 官网,如下图所示: 由于我们初期只在windows上

北京筑龙吴英礼:ChatGPT对采购与招标数字化的影响

2月25日下午&#xff0c;平台经济学沙龙&#xff08;第八期&#xff09;在清华大学互联网产业研究院成功举办。本期沙龙以“ChatGPT对招标采购的影响”为主题&#xff0c;由清华大学互联网产业研究院平台经济课题组组长、中国招标投标公共服务平台原总经理、首席经济学家平庆忠…

Leetcode19. 删除链表的倒数第n个结点

一、题目描述&#xff1a; 给你一个链表&#xff0c;删除链表的倒数第 n 个结点&#xff0c;并且返回链表的头结点。 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,4,5], n 2输出&#xff1a;[1,2,3,5] 示例 2&#xff1a; 输入&#xff1a;head [1], n 1输出&#x…

Android开发八股文,Android也有自己的八股文了

前言别的行业都有自己的八股文&#xff0c;凭什么Android没有。2023春招即将来临&#xff0c;很多同学会问 Android开发的面试题有必要背吗&#xff1f;我的回答是&#xff1a;很有必要。你可以讨厌这种模式&#xff0c;但你一定要去背&#xff0c;因为不背你就进不了大厂。国内…

SSL/TLS协议信息泄露漏洞(CVE-2016-2183)调查解决

目录结构前言测试回馈漏洞介绍漏洞解决参考文献前言 产品测试阶段出现“SSL/TLS协议信息泄露漏洞&#xff08;CVE-2016-2183&#xff09;”&#xff0c;解决过程记录如下 测试回馈 建议&#xff1a;避免使用IDEA、DES和3DES算法 1、OpenSSL Security Advisory [22 Sep 2016] …

P02 滴水逆向1月4号公开课

公开课进制转换汇编如何工作的逻辑运算二进制逻辑运算|| 运算 和 | 运算&& 运算 和 & 运算^(异或) 运算! (非)运算左移二进制逻辑运算应用加法运算运算与电脑硬件进行加密寄存器内存寻址范围使用内存寻址公式堆栈变形的艺术进制转换 二进制十六进制如何转换的 汇…

k8s学习之路 | Day17 k8s 工作负载

文章目录工作负载的定义工作负载资源分类工作负载的定义 官方参考链接&#xff1a;https://kubernetes.io/docs/concepts/workloads/ A workload is an application running on Kubernetes. Whether your workload is a single component or several that work together, on K…

已解决ERROR: Failed building wheel for opencv-python-headless

已解决ERROR: Failed building wheel for opencv-python-headless Failed to build opencv-python-headless ERROR: Could not build wheels for opencv-python-headless, which is required to install pyproject.toml-based projects报错信息亲测有效 文章目录报错问题报错翻…

多任务学习概述

文章目录前言1 文章信息2 背景、目的、结论2.1 背景2.1.1 多任务的类型分类2.1.1.1 相关任务的分类2.1.1.2 将输入变输出的逆多任务学习2.1.1.3 对抗性多任务学习2.1.1.4 辅助任务提供注意力特征的多任务学习2.1.1.5 附加预测性辅助任务的多任务学习3 内容与讨论3.1 多任务学习…

HTML+ CSS 笔记

CSS1. 外边距合并 margin2. 浮动和清除浮动3. CSS 属性书写顺序(重点)4. 页面布局分析5. 定位定位模式子绝父相&#xff1a;z-index定位的拓展6. 元素的显示与隐藏7. 精灵图 sprites8. 字体图标 iconfont9. 居中10. 文字溢出处理情况11.HTML5 和CSS3 新增标签12.品优购1. 项目规…

多数据库管理工具哪家强?ChatGPT点评,第一位并不是Navicat

SQL逐渐成为职场必备的编程语言&#xff0c;相信大家都不陌生。SQL是一种结构化查询语言&#xff0c;是用于数据库之间通信的编程语言。每个数据库都有着自己独特的访问规则&#xff0c;但大体上是遵循SQL标准。 因此&#xff0c;辗转于不同的数据库之间&#xff0c;开发者或D…

Codeforces Round #852 (Div. 2) D. Moscow Gorillas

传送门 题意&#xff1a; 给你两个长度为n的排列p&#xff0c;q&#xff0c;同时定义mex表示为在mex中&#xff0c;缺少的最小正整数&#xff1a;如 mex[1&#xff0c;3]2&#xff0c;mex[1,2,3]4。 问你有多少个l,r使mex([])mex([])。 思路 &#xff1a; 可以从mex1,2,3,4,..n…

Word 行中公式与文字对不齐的解决方案

Word 行中公式与文字对不齐的解决方案问题-公式与文字不对齐解决方案简述版(亲测)解决方案详细版(亲测)参考方案问题-公式与文字不对齐 由于复制或其它误操作&#xff0c;可能会出现Word一行中公式与文字并不对齐的问题&#xff0c;有的公式会偏上或偏下&#xff0c;如下图所示…

【Linux内核】内核编程概述

1.GNU GNU计划和自由软件基金会&#xff08;FSF&#xff09;是由Richard M.Stallman于1984年创办的&#xff0c;GNU是“GNU’s Not UNIX”的缩写。到20世纪90年代初&#xff0c;GNU项目已经开发出许多高质量的免费软件&#xff0c;其中包括emacs编辑系统、bash shell程序、gcc…

PIGOSS BSM 信创运维之国产数据库监控

随着信创的推进&#xff0c;为国产数据库厂商提供了发展沃土&#xff0c;国产厂商扮演者越来越重要的角色。国产数据库以达梦、人大金仓、南大通用、高斯数据库GaussDB、万里开源为代表&#xff0c;而新兴厂商则利用分布式进行换道超车&#xff0c;以PingCAP、巨杉数据库、星环…

JAVA对象

目录 1.组成 1.1.概述 1.2.对象头 1.3.实例数据 1.4.对齐填充字节 2.实例化流程 1.组成 1.1.概述 JAVA对象由三部分组成&#xff1a; 对象头实例数据对齐填充字节1.2.对象头 对象头里面包含3部分主要内容&#xff1a; markwordklass数组长度&#xff08;只有数组对象…

StreamYOLO 代码阅读记录

目录 一、安装配置环境 二、运行 train.py文件遇到的问题 问题一 问题二 问题三 问题四 问题五 问题六 问题七 问题八 二、debug记录 1、 args 2、optimizer 3、ckpt 4、self.seq_dirs 5、self._classes 6、im_ann 7、annotations 8、obj 9、self.train_loade…

Qt——拖拽

Qt的拖拽可以按字面意思分为拖和拽两部分。一般来说我们常见的拖拽分别由两个程序合作完成。例如我们经常把桌面的文件拖拽进其他目录&#xff1a; ​ 编辑拖拽方可以发起多个拖拽类型&#xff08;复制、链接、移动等&#xff09;&#xff0c;接收方可以选择接受其中某个类型或…

《分布式技术原理与算法解析》学习笔记Day25

负载均衡 负载均衡是分布式可靠性中非常关键的一个问题&#xff0c;它在一定程度上反映了分布式系统对业务处理的能力。 什么是负载均衡&#xff1f; 负载均衡可以分为两种&#xff1a; 请求负载均衡&#xff0c;即将用户的请求均衡的分发到不同的服务器进行处理。数据负载…