Java中的各种list有什么区别及list和set区别详析

 更新时间:2026年06月02日 10:01:10   作者:y = xⁿ  
List是Java集合框架中使用频率比较高的一个接口,它的特点是存储的数据是有序的并且数据是可以重复的,这篇文章主要介绍了Java中各种list有什么区别及list和set区别的相关资料,需要的朋友可以参考下

Java 中 List 的几种实现

List 是 有序、可重复 的集合接口
常见实现:ArrayList、LinkedList、Vector、Stack、CopyOnWriteArrayList

ArrayList

底层结构

  • 动态数组
  • 初始容量:10
  • 扩容机制

新容量 = 旧容量 + 旧容量 / 2 (1.5 倍)

时间复杂度

操作复杂度说明
随机访问 get(i)O(1)数组下标
尾部 addO(1) 均摊扩容时 O(n)
中间插入/删除O(n)元素整体移动

线程安全

  • 非线程安全

解决方案:Collections.synchronizedList;CopyOnWriteArrayList

LinkedList

底层结构

  • 双向链表
  • 每个节点:prev | item | next

时间复杂度

操作复杂度说明
get(i)O(n)要遍历
头/尾插入删除O(1)指针操作
中间插入O(n)找位置

Vector

底层结构

-动态数组(和 ArrayList 类似)

关键区别

  • 线程安全
  • 方法都加了 synchronized

问题

  • 性能差(锁太重)
  • 已被淘汰

Stack

继承关系

Stack extends Vector

特点

  • 后进先出(LIFO)
  • 线程安全(继承 Vector)

CopyOnWriteArrayList(并发重点)

底层思想

  • 写时复制

特点

方面说明
线程安全
读性能非常高
写性能较差(复制数组)
迭代不会抛 ConcurrentModificationException

对比总结

实现底层线程安全适合场景
ArrayList动态数组查询多
LinkedList双向链表头尾操作多
Vector动态数组淘汰
Stack淘汰
CopyOnWriteArrayList数组复制并发读多

面试常见问题

Q1:ArrayList 和 LinkedList 区别?

ArrayList:数组,查询快,插入慢
LinkedList:链表,头尾操作快,随机访问慢

Q2:为什么 ArrayList 不是线程安全?

add / remove 过程中可能发生:
扩容
覆盖
数据丢失

Q3:CopyOnWriteArrayList 为什么读快?

读操作不加锁
始终读的是稳定数组快照

Q4:为什么不推荐 Vector / Stack?

synchronized 粒度太大
性能差
有更好的并发方案

总结

ArrayList 查得快,LinkedList 插得快(头尾),
并发读多用 CopyOnWrite,Vector Stack 已淘汰

一、关于List和Set

List vs Set

对比点ListSet
是否有序✅ 有序(按插入顺序)❌ 大多无序(LinkedHashSet 例外)
是否允许重复✅ 允许❌ 不允许
是否有下标✅ 有(get(i)❌ 没有
常见实现ArrayListLinkedListHashSetLinkedHashSetTreeSet
典型用途保存“有顺序、可重复”的数据保存“去重”的数据

一句话记忆:

List = 有顺序 + 可重复
Set = 去重

二、List

什么是List?

  • 像数组的升级版
  • 能按顺序存
  • 能存重复元素
  • 能通过下标访问
List<Integer> list = new ArrayList<>();
list.add(10);
list.add(10);
list.add(20);

System.out.println(list); // [10, 10, 20]
System.out.println(list.get(1)); // 10

三、Set

什么是Set?

  • 天然去重
  • 关心你插入顺序
  • 没有下标
Set<Integer> set = new HashSet<>();
set.add(10);
set.add(10);
set.add(20);

System.out.println(set); // [10, 20]

Set 是怎么判断“重复”的?

靠 hashCode() + equals()

  1. 先算 hashCode
  2. 再用 equals 比较

四、List vs Set 核心区别

List 和 Set 都是 Collection 的子接口,主要区别在于是否允许重复和是否有顺序。List 允许重复元素并且有下标,适合顺序存储;Set 不允许重复元素,主要用于去重。

五、一句话总结

List:顺序 + 重复 + 下标
Set:去重 + 无下标 + 基于 equals/hashCode

到此这篇关于Java中各种list有什么区别及list和set区别详析的文章就介绍到这了,更多相关Java中list和set内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • MyBatis框架底层的执行原理源码解析

    MyBatis框架底层的执行原理源码解析

    这篇文章主要介绍了MyBatis框架底层的执行原理源码解析,本文通过图文实例代码相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-06-06
  • 关于@Scheduled注解的任务为什么不执行的问题

    关于@Scheduled注解的任务为什么不执行的问题

    这篇文章主要介绍了关于@Scheduled注解的任务为什么不执行的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-09-09
  • Guava Retryer实现接口重试的示例

    Guava Retryer实现接口重试的示例

    本文主要介绍了Guava Retryer实现接口重试的示例,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-12-12
  • 使用SpringJPA 直接实现count(*)

    使用SpringJPA 直接实现count(*)

    这篇文章主要介绍了SpringJPA 直接实现count(*),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • springboot 使用QQ邮箱发送邮件的操作方法

    springboot 使用QQ邮箱发送邮件的操作方法

    这篇文章主要介绍了springboot使用QQ邮箱发送邮件功能,本文通过实例图文相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-10-10
  • 使用java判断输入年份是否为闰年完整代码

    使用java判断输入年份是否为闰年完整代码

    闰年的引入确保了我们的日历与地球运行轨道的对齐,使得时间的计算更加准确,在编程中判断给定年份是否为闰年是一项常见的任务,这篇文章主要给大家介绍了关于使用java判断输入年份是否为闰年的相关资料,需要的朋友可以参考下
    2023-10-10
  • SpringCloudAlibaba整合服务网关GateWay教程

    SpringCloudAlibaba整合服务网关GateWay教程

    文章介绍了Spring Cloud Gateway和Zuul的区别,并详细描述了如何使用Gateway进行服务转发和自定义TokenFilter实现参数拦截访问
    2026-01-01
  • 浅析Java方法传值和传引用问题

    浅析Java方法传值和传引用问题

    这篇文章主要是对Java方法传值和传引用问题进行了详细的介绍,需要的朋友可以过来参考下,希望对大家有所帮助
    2013-12-12
  • Java Web实现自动登陆功能

    Java Web实现自动登陆功能

    这篇文章主要为大家详细介绍了Java Web实现自动登陆功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08
  • springboot中使用redis并且执行调试lua脚本

    springboot中使用redis并且执行调试lua脚本

    今天有个项目需要使用redis,并且有使用脚本的需求,本文主要介绍了springboot中使用redis并且执行调试lua脚本,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-04-04

最新评论