Java集合之List(超详细)

List是Java集合框架中一个非常重要的接口,它代表了一个有序的集合,允许元素重复,并且可以按照插入的顺序进行访问。

我们先来看看List在集合中的位置:

在这里插入图片描述

List是单列集合接口Collection下的一个分支,另两个分支是SetQueue,三者的区别:

  • List集合中的元素是有序的、可重复的
  • Set集合中的元素无需、不可重复
  • Qeue集合中的元素遵循先进先出的规则

一,List的继承体系

List接口继承自Collection接口,位于java.util包中。

List是有序集合的抽象表示,Java发展至今,List体系已经非常庞杂。

在这里插入图片描述

从上图看出,JDK中,直接或间接继承List接口的有80个类。当然,我们无需一一学习,通过三个常用的实现类的学习掌握原理即可:ArrayList、LinkedList和Vector。

  • ArrayList:基于动态数组实现,提供了快速的随机访问。
  • LinkedList:基于双向链表实现,擅长插入和删除操作,尤其是表头和表尾的操作。
  • Vector:早期版本的线程安全列表,与ArrayList相似,但现在多被ArrayList取代,因同步开销较大。

二,List的常用操作及代码示例

1,创建List实例

import java.util.*;public class ListDemo {public static void main(String[] args) {List<String> arrayList = new ArrayList<>(); // 创建ArrayList实例List<String> linkedList = new LinkedList<>(); // 创建LinkedList实例}
}

2,增加元素

  • add(E element):在列表末尾添加元素。
  • add(int index, E element):在指定位置插入元素。
arrayList.add("Apple");
linkedList.add(0, "Banana"); // 在首位插入

3,访问元素

  • 通过索引访问:get(int index)
  • 遍历:使用for-each循环或迭代器Iterator。
System.out.println(arrayList.get(0));
for (String fruit : arrayList) {System.out.println(fruit);
}

4,修改元素

  • 使用set(int index, E element)方法替换指定位置的元素。
arrayList.set(0, "Orange");

5,删除元素

  • remove(int index):根据索引删除。
  • remove(Object o):根据元素删除第一个匹配项。
arrayList.remove(0);
arrayList.remove("Orange");

6,判断与查找

  • contains(Object o):判断是否包含某元素。
  • indexOf(Object o):返回元素第一次出现的索引,未找到返回-1。
  • lastIndexOf(Object o):List集合中的元素可重复,返回元素最后一次出现的索引,未找到返回-1。
boolean hasApple = arrayList.contains("Apple");
int index = arrayList.indexOf("Apple");
int index = arrayList.lastIndexOf("Apple");

7,大小与清空

  • size():返回列表大小。
  • clear():清空列表。
int size = arrayList.size();
arrayList.clear();

8,List集合的遍历

Java List 接口提供了多种方式来遍历其中的元素,以下是三种常见的遍历方式,每种方式都有相应的代码示例。

① 使用 for-each 循环

这是最简洁也是最常用的遍历方式,适用于Java 5及以后的版本。通过for-each循环,可以直接遍历List中的每个元素,而无需手动管理索引。

代码示例

import java.util.ArrayList;
import java.util.List;public class ListTraversal {public static void main(String[] args) {List<String> fruits = new ArrayList<>();fruits.add("Apple");fruits.add("Banana");fruits.add("Cherry");// 使用for-each循环遍历for (String fruit : fruits) {System.out.println(fruit);}}
}
并发修改异常ConcurrentModificationException

**注意,**使用for循环时,有可能会出现并发修改异常ConcurrentModificationException,如下面的例子,假设你有一个任务是遍历一个List,检查其中的元素,如果满足某个条件,就从List中删除该元素。

List<String> list = new ArrayList<>(Arrays.asList("A", "B", "C", "D"));for (String item : list) {if ("B".equals(item)) {list.remove(item); // 这里会抛出ConcurrentModificationException}
}

Java集合框架中的许多类,如ArrayListLinkedList,为了检测到并发修改,使用了所谓的“快速失败”机制:

  • ①当迭代器创建之后,集合会维护一个名为modCount的字段来记录集合的修改次数
  • ②每当集合通过迭代器之外的方式(如直接调用addremove方法)发生修改时,modCount就会递增
  • ③迭代器在每次调用nexthasNext等方法时,都会检查这个计数器是否发生变化,如果发现modCount不等于它内部记录的初始修改次数,就会抛出ConcurrentModificationException

据此分析,上面的代码示例之所以会报错,是因为:

  • ①使用for循环时会创建迭代器,迭代器会缓存当前modCount的值
  • ②在循环中使用了remove,modCount的值发生了变化
  • ③下一循环时,迭代器会检查缓存的modCount值与真实的modCount值是否一致,不一致就会抛出错误
并发修改异常的解决办法
  • ①使用迭代器的remove方法

正确的做法是在迭代过程中使用迭代器的remove方法来删除元素,因为迭代器的remove方法会在删除元素后同时更新内部的修改计数,以保持一致性。

Iterator<String> iterator = list.iterator();
while (iterator.hasNext()) {String item = iterator.next();if ("B".equals(item)) {iterator.remove(); // 正确的删除方式}
}
  • ② 使用CopyOnWriteArrayList

对于多线程环境下的并发修改问题,可以考虑使用CopyOnWriteArrayList。这是一种线程安全的List实现,它通过在每次修改时创建集合的副本来避免并发修改异常,适合读多写少的场景。

List<String> list = new CopyOnWriteArrayList<>(Arrays.asList("A", "B", "C", "D"));
list.removeIf("B"::equals); // 线程安全的删除操作
  • ② 使用迭代器 Iterator

迭代器是一种更通用的遍历集合的方法,适用于所有实现了Iterable接口的集合,包括List。通过调用List的iterator()方法获取Iterator对象,然后使用hasNext()next()方法进行遍历。

代码示例

import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;public class ListTraversal {public static void main(String[] args) {List<String> fruits = new ArrayList<>();fruits.add("Apple");fruits.add("Banana");fruits.add("Cherry");// 使用迭代器遍历Iterator<String> iterator = fruits.iterator();while (iterator.hasNext()) {String fruit = iterator.next();System.out.println(fruit);}}
}
  • ③ 使用Java 8的Stream API

从Java 8开始,可以使用Stream API来遍历和处理集合中的元素,这种方式更加灵活,支持函数式编程风格。

代码示例

import java.util.ArrayList;
import java.util.List;public class ListTraversal {public static void main(String[] args) {List<String> fruits = new ArrayList<>();fruits.add("Apple");fruits.add("Banana");fruits.add("Cherry");// 使用Stream API遍历fruits.stream().forEach(System.out::println);}
}
④ 总结
  • for-each循环:简洁易读,最适合日常使用。
  • Iterator:提供了更多的控制权,比如在遍历时移除元素,但在大多数情况下不如for-each方便。
  • Stream API:功能强大,支持复杂的集合处理和并行处理,适合进行复杂的聚合操作和过滤操作。

9,List集合的排序

在Java中,对List集合中的元素进行排序可以通过多种方式实现,主要依赖于java.util.Collections类和List接口本身提供的排序方法。下面我将介绍几种常见的排序方法,并提供相应的代码示例。

  • ① 使用Collections.sort()方法

这是最直接的方式,适用于实现了Comparable接口的元素列表,进行自然排序。

代码示例(自然排序):

import java.util.*;class Person implements Comparable<Person> {String name;int age;Person(String name, int age) {this.name = name;this.age = age;}@Overridepublic int compareTo(Person other) {return Integer.compare(this.age, other.age); // 按年龄排序}@Overridepublic String toString() {return name + " " + age;}
}public class SortListExample {public static void main(String[] args) {List<Person> people = new ArrayList<>();people.add(new Person("Tom", 25));people.add(new Person("Jerry", 20));people.add(new Person("Bob", 30));Collections.sort(people);for (Person person : people) {System.out.println(person);}}
}
  • ②使用Collections.sort()方法和自定义Comparator

如果列表中的元素没有实现Comparable接口,或者你想根据不同的规则进行排序,可以提供一个Comparator

代码示例(自定义比较器排序):

import java.util.*;public class SortListExample {public static void main(String[] args) {List<String> names = Arrays.asList("Alice", "Bob", "Charlie");Collections.sort(names, new Comparator<String>() {@Overridepublic int compare(String s1, String s2) {return s2.compareTo(s1); // 倒序排序}});names.forEach(System.out::println); // 输出:Charlie, Bob, Alice}
}
  • ③使用List自带的sort()方法

从Java 8开始,List接口直接提供了sort()方法,它同样接受Comparator来控制排序逻辑。

代码示例

import java.util.*;public class SortListExample {public static void main(String[] args) {List<Integer> numbers = Arrays.asList(3, 1, 4, 1, 5, 9);numbers.sort(Integer::compareTo); // 自然排序System.out.println(numbers); // 输出:[1, 1, 3, 4, 5, 9]numbers.sort(Collections.reverseOrder()); // 倒序排序System.out.println(numbers); // 输出:[9, 5, 4, 3, 1, 1]}
}
  • ④小结
  • 使用Collections.sort()适用于不支持Lambda表达式的较早Java版本。
  • 从Java 8开始,直接使用List的sort()方法配合Lambda表达式或方法引用来实现排序更为简洁。
  • 通过自定义Comparator,可以灵活地控制排序逻辑,适应不同的排序需求。

三,不同List实现的底层原理及区别

ArrayList

  • 底层原理:基于可变大小的数组实现,数组扩容时会创建新数组并复制旧数据。
  • 适用场景:当需要频繁查询元素,且元素数量变化不大时效率高。

LinkedList

  • 底层原理:每个元素都是一个节点,包含前驱和后继节点的引用,形成双向链表。
  • 适用场景:适合于频繁的插入和删除操作,尤其是在列表的开始或结尾。

Vector

  • 底层原理:与ArrayList相似,但Vector是线程安全的,通过在关键方法上加锁实现。
  • 区别与注意事项:由于同步操作,Vector在多线程环境下更安全,但并发访问时性能较低。现代开发中,推荐使用Collections.synchronizedList(List<T> list)CopyOnWriteArrayList作为替代。

通过上述内容,我们不仅了解了List接口的继承体系、常用操作,还深入探讨了ArrayList、LinkedList和Vector这三种常见实现的底层原理及其应用场景。掌握这些知识,将有助于在实际开发中更加高效、灵活地使用List集合。

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

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

相关文章

【Redis数据库百万字详解】数据类型

文章目录 一、字符串类型概述1.1、数据类型1.2、字符串简介1.3、字符串应用场景 二、字符串命令三、哈希类型概述3.1、哈希介绍3.2、哈希类型应用场景3.3、哈希命令 四、列表类型概述4.1、列表简介4.2、使用场景4.3、列表命令 五、集合概述5.1、集合简介5.2、使用场景5.3、集合…

[大师C语言(第二十一篇)]C语言字节对齐技术详解

引言 在计算机系统中&#xff0c;内存对齐是一种非常重要的技术。它指的是数据在内存中的存放位置与内存地址之间的关系。C语言作为一种高级编程语言&#xff0c;提供了丰富的内存对齐操作&#xff0c;使得程序员可以灵活地控制数据在内存中的布局。本文将深入探讨C语言对齐背…

JavaScript中,ToPrimitive的操作把对象转化为原始值

在JavaScript中&#xff0c;ToPrimitive是一个抽象操作&#xff0c;不是一个实际的方法。ToPrimitive操作用于将对象转换为原始值&#xff08;例如&#xff0c;字符串、数字或布尔值&#xff09;。这个操作通常在需要原始值的情况下自动执行&#xff0c;例如在比较或算术运算中…

网络工程从头做-1

网络工程从头做-1 自下而上&#xff0c;从接入交换机开始网络的配置和规划 实验拓扑&#xff1a; 实验步骤&#xff1a; 1.完成基本配置 1.1 PC端IP地址信息配置略 1.2 接入层交换机S1配置 [Huawei]sys S1 [S1]undo in [S1]vlan b 10 20 [S1]int e0/0/1 [S1-Ethernet0/0/1]p l…

k8s怎么监听自定义资源的变更?(2)

接上一篇当生成下面代码之后怎么去使用呢&#xff1f; 1.生成crd文件 这里我们通过kubebuilder的一个子项目 controller-gen 来生成crd文件 https://github.com/kubernetes-sigs/controller-tools curl -L -o https://github.com/kubernetes-sigs/controller-tools; go ins…

48、Flink 的 Data Source API 详解

a&#xff09;概述 本节将描述 FLIP-27 中引入的新 Source API 的主要接口。 b&#xff09;Source Source API 是一个工厂模式的接口&#xff0c;用于创建以下组件。 Split EnumeratorSource ReaderSplit SerializerEnumerator Checkpoint Serializer 此外&#xff0c;Sou…

D-Day 上海站回顾丨以科技赋能量化机构业务

5月31日下午&#xff0c;DolphinDB 携手光大证券&#xff0c;在上海成功举办 D-Day 行业交流会。三十余位来自私募机构的核心策略研发、量化交易员、数据分析专家们齐聚现场&#xff0c;深入交流量化投研交易过程中的经验、挑战及解决方案。 DolphinDB 赋能机构业务平台 来自光…

1877java项目建设平台管理系统Myeclipse开发mysql数据库web结构java编程计算机网页项目

一、源码特点 java 项目建设平台管理系统 是一套完善的web设计系统&#xff0c;对理解JSP java编程开发语言有帮助采用了java设计&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统采用web模式&#xff0c;系统主要采用B/S模式开 发。开发环境为TOMCAT7.0,Myeclipse8.…

java表实体 蛇形转驼峰 正则匹配替换

java表实体 蛇形转驼峰 正则匹配替换 1.匹配寻找正则&#xff1a;([a-z])_([a-z])2.替换结果正则&#xff1a;$1\U$2\E效果如下图所示&#xff1a;

Python第二语言(三、Python函数def)

目录 1. Python函数&#xff08;def 函数名():&#xff09; 1.1 sorted对容器进行排序&#xff1a;无法指定排序规则 1.2 sort对容器自定义排序&#xff1a;可以指定排序规则 1.3 获取变量长度函数&#xff08;len&#xff09; 1.4 函数的定义 1.5 函数-传参定义 1.6 函…

如何使用 Systemd 和 Nginx 部署 Node.js 应用程序

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。 介绍 在将 Web 应用部署到 Droplet 时,可能会诱人地简单地使用与开发中相同的设置,即在终端中运行“ruby app.rb”或“node server.js”来启动服务器。这样做简单易行…

C#-for循环语句

for循环语句 语法: for(初始化变量; 判断条件; 增量表达式) { // 内部代码 } 第一个空(初始表达式): 一般用来声明一个临时的局部变量 用来计数第二个空(条件表达式): 表明进入循环的条件 一个bool类型的值(bool类型 条件表达式 逻辑运算符)第三个空(增量表达式): 使用第一个空…

Python怎么翻译:探索Python在翻译领域的无限可能

Python怎么翻译&#xff1a;探索Python在翻译领域的无限可能 Python&#xff0c;这门强大而灵活的编程语言&#xff0c;已经在众多领域展现了其独特的魅力。然而&#xff0c;当谈到翻译这一领域时&#xff0c;许多人可能会感到困惑&#xff1a;Python怎么能用于翻译呢&#xf…

OpenCV如何判断一张图片是否有过高的明暗变化

操作系统&#xff1a;ubuntu22.04OpenCV版本&#xff1a;OpenCV4.9IDE:Visual Studio Code编程语言&#xff1a;C11 前言 判断一张图片是否有过高的明暗变化&#xff0c;可以通过分析图像的亮度分布一致性来实现。一种常见的做法是计算图像的亮度标准差&#xff08;Standard …

免费,C++蓝桥杯等级考试真题--第7级(含答案解析和代码)

C蓝桥杯等级考试真题--第7级 答案&#xff1a;D 解析&#xff1a;步骤如下&#xff1a; 首先&#xff0c;--a 操作会使 a 的值减1&#xff0c;因此 a 变为 3。判断 a > b 即 3 > 3&#xff0c;此时表达式为假&#xff0c;因为 --a 后 a 并不大于 b。因此&#xff0c;程…

ESP32-C3模组上跑通NVS(4)

接前一篇文章&#xff1a;ESP32-C3模组上跑通NVS&#xff08;3&#xff09; 本文内容参考&#xff1a; 非易失性存储库 - ESP32 - — ESP-IDF 编程指南 latest 文档 ESP32-C3入门教程 基础篇&#xff08;八、NVS — 非易失性存储库的使用&#xff09;_esp入门教学-CSDN博客 …

STM32 启用指令缓存 HAL_ICACHE_Enable

函数在 STM32 的 HAL&#xff08;硬件抽象层&#xff09;库中通常用于启用指令缓存&#xff08;I-Cache&#xff09;。以下是该函数的主要功能&#xff1a; 启用指令缓存&#xff1a; 当调用 HAL_ICACHE_Enable 函数时&#xff0c;STM32 的 Cortex-M 处理器&#xff08;特别是…

ElementUI的Table组件在无数据情况下让“暂无数据”文本居中显示

::v-deep .el-table__empty-block {width: 100%;min-width: 100%;max-width: 100%; }

如何在npm上发布自己的包

如何在npm上发布自己的包 npm创建自己的包 一、一个简单的创建 1、创建npm账号 官网&#xff1a;https://www.npmjs.com/创建账号入口&#xff1a;https://www.npmjs.com/signup 注意&#xff1a;需要进入邮箱验证 2、创建目录及初始化 $ mkdir ufrontend-test $ cd ufron…

今日科普:了解、预防、控制高血压

高血压&#xff0c;常被称为“隐形的健康威胁”&#xff0c;许多患者可能在毫无预警的情况下发病&#xff0c;且患病率逐年攀升&#xff0c;同时患者群体逐渐年轻化&#xff0c;高血压虽然难以根治&#xff0c;但并不可怕&#xff0c;真正可怕的是血压长期居高不下&#xff0c;…