泪伤荡的编程指南 泪伤荡的编程指南
首页
  • 基础篇
  • 集合篇
  • 并发篇
  • JVM篇
  • 新特性
  • 进阶篇
  • 网络
  • 操作系统
  • 数据结构与算法
  • 硬件
  • 基础篇
  • MySql
  • Oracle
  • PostgreSQL
  • 达梦
  • Redis
  • Mongodb
  • Hive
  • 数据库比较
  • Spring
  • SpringMvc
  • SpringBoot
  • Hibernate
  • iBatis
  • Mybatis
  • Mybatis-plus
  • Mybatis-plus-join
  • 各个框架对比
  • UML画图
  • 设计须知
  • 开发流程
  • 开发理论
  • 架构体系
  • 设计模式
  • 开源知识
  • 分布式解决方案
  • SpringCloud
  • API网关
  • 注册中心
  • 配置中心
  • 服务调用
  • 分布式事务
  • 消息队列
  • 调度作业
  • 链路追踪
  • 服务保障
  • 搜索引擎Elk
  • 安全框架
  • 监控体系
  • 部署容器
  • Netty
  • Tomcat
  • Nginx
  • 图片云存储
  • 云存储
  • 虚拟机Linux
  • 项目部署
  • 容器部署
  • 开发工具篇
  • 工具库篇
  • 开发技巧篇
  • 工具类系列
  • Bug记录仓库
  • 随笔
  • HTML与CSS
  • JS学习
  • Vue3入门
  • Vue3进阶
  • 黑马Vue3
  • 视频网站
  • 音乐网站
  • 商城网站
  • 论坛网站
  • scrm项目
  • Yudao-cloud
  • RuoYi-Vu-cloud
  • 博客搭建
  • 网站收藏箱
  • 断墨寻径摘录
  • 费曼学习法
  • Java术语
  • 命名英语
  • 业务英语
  • 表字段英语
  • 包名英语
Github (opens new window)
首页
  • 基础篇
  • 集合篇
  • 并发篇
  • JVM篇
  • 新特性
  • 进阶篇
  • 网络
  • 操作系统
  • 数据结构与算法
  • 硬件
  • 基础篇
  • MySql
  • Oracle
  • PostgreSQL
  • 达梦
  • Redis
  • Mongodb
  • Hive
  • 数据库比较
  • Spring
  • SpringMvc
  • SpringBoot
  • Hibernate
  • iBatis
  • Mybatis
  • Mybatis-plus
  • Mybatis-plus-join
  • 各个框架对比
  • UML画图
  • 设计须知
  • 开发流程
  • 开发理论
  • 架构体系
  • 设计模式
  • 开源知识
  • 分布式解决方案
  • SpringCloud
  • API网关
  • 注册中心
  • 配置中心
  • 服务调用
  • 分布式事务
  • 消息队列
  • 调度作业
  • 链路追踪
  • 服务保障
  • 搜索引擎Elk
  • 安全框架
  • 监控体系
  • 部署容器
  • Netty
  • Tomcat
  • Nginx
  • 图片云存储
  • 云存储
  • 虚拟机Linux
  • 项目部署
  • 容器部署
  • 开发工具篇
  • 工具库篇
  • 开发技巧篇
  • 工具类系列
  • Bug记录仓库
  • 随笔
  • HTML与CSS
  • JS学习
  • Vue3入门
  • Vue3进阶
  • 黑马Vue3
  • 视频网站
  • 音乐网站
  • 商城网站
  • 论坛网站
  • scrm项目
  • Yudao-cloud
  • RuoYi-Vu-cloud
  • 博客搭建
  • 网站收藏箱
  • 断墨寻径摘录
  • 费曼学习法
  • Java术语
  • 命名英语
  • 业务英语
  • 表字段英语
  • 包名英语
Github (opens new window)
  • 基础篇

    • JavaSE - 初始 Java
    • JavaSE - 注释
    • JavaSE - 标识符
    • JavaSE - 数据类型
    • JavaSE - 类型转换
    • JavaSE - 变量和常量
    • JavaSE - 运算符
    • JavaSE - 包机制
    • JavaSE - JavaDoc
    • JavaSE - 方法
    • JavaSE - 用户交互 Scanner
    • JavaSE - 顺序结构
    • JavaSE - 选择结构
    • JavaSE - 循环结构
    • JavaSE - Break 和 Continue
    • JavaSE - 数组
    • JavaSE - 异常机制
    • JavaSE - 面向对象
    • JavaSE - 常用类
    • JavaSE - IO流
    • 思考 - 如何快速掌握一门语言
  • 集合篇

    • Java集合概述
    • List接口
      • ArrayList 和 Vector 的区别
      • ArrayList 与 LinkedList 区别
        • 具体实现参考
        • 什么是迭代器
    • Set接口
    • Queue接口
    • Map接口
  • 并发篇

    • Java并发基础小结
    • 锁详解
    • Synchronized和Volatile的使用与区别
    • 线程池详解
    • CompletableFuture学习
    • Java内存管理总结
  • JVM篇

    • JVM基础入门
    • JVM常问
  • 新特性

    • JDK5

      • 网站内容
      • 手动填充
    • JDK8

      • Java8 新特性
      • JDK新特性
    • JDK11

      • 网站内容
      • 手动填充
    • JDK17

      • 网站内容
      • 手动填充
    • JDK21

      • 网站内容
      • 手动填充
    • JDK25

      • 网站内容
      • 手动填充
  • 进阶篇

    • Java程序设计概述
    • Java程序设计环境
    • Java的基本程序设计结构
    • 对象与类
    • 继承
    • 接口、lambda表达式与内部类
    • 异常、断言和日志
    • 泛型程序设计
    • 集合
    • 图形用户界面程序设计
    • Swing用户界面组件
    • 并发
    • 附录-Java关键字
    • Java泛型核心知识总结
    • Java反射详解
    • Java SPI机制详解
  • Java
  • 集合篇
泪伤荡
2023-08-10
目录

List接口

# List 集合详解

# ArrayList 和 Vector 的区别

  1. 线程安全性:
    • Vector 是线程安全的,它的方法都是同步的,多个线程可以同时访问和修改 Vector 对象;
    • 而 ArrayList 是非线程安全的,它的方法不是同步的,多个线程访问和修改同一个 ArrayList 对象可能会导致数据竞争和并发访问的问题。
  2. 扩容方式:Vector 和 ArrayList 在扩容时采用不同的策略。Vector 在扩容时会增加一倍的容量,而 ArrayList 则会增加 50% 的容量。
  3. 性能:
    • 由于 Vector 是线程安全的,它在并发访问时需要进行同步操作,因此性能相对较低;
    • 而 ArrayList 在单线程环境下的性能较好,但在多线程环境下需要使用同步机制来保证线程安全。
  4. 初始容量:当创建 Vector 或 ArrayList 对象时,可以指定它们的初始容量。Vector 的默认初始容量为 10,而 ArrayList 的默认初始容量为 0。在实际使用中,可以根据数据量和性能需求等因素,选择合适的初始容量。

总结

  • Vector 和 ArrayList 都是动态数组的实现,
  • 但 Vector 是线程安全的,扩容方式和性能相对较差,初始容量为 10;
  • 而 ArrayList 是非线程安全的,扩容方式和性能相对较好,初始容量为 0。
  • 在实际使用中,应根据具体需求选择合适的动态数组实现。

# ArrayList 与 LinkedList 区别

  1. 底层数据结构:

    • ArrayList 底层采用数组实现,
    • 而 LinkedList 底层采用链表实现。
  2. 随机访问(相当于查询)和插入/删除操作的性能:

    • 由于 ArrayList 的底层实现是数组,因此随机访问的性能较好,时间复杂度为 O(1);而插入/删除操作需要移动其他元素,时间复杂度为 O(n)。

    • 相反,LinkedList 的底层实现是链表,因此插入/删除操作的性能较好,平均时间复杂度为 O(1);而随机访问需要遍历链表,时间复杂度为 O(n)。

      链表进行【插入/删除操作】时,在最坏的情况下,目标位置在链表的中间,需要遍历链表来找到目标位置,导致时间复杂度为 O(n)。

  3. 内存占用:

    • 由于 ArrayList 底层采用数组实现,因此需要预先分配一定大小的连续内存空间,因此可能会浪费一些内存空间;
    • 而 LinkedList 的底层采用链表实现,因此每个节点可以分布在不同的内存空间,内存利用率相对较高。
  4. 迭代器的性能:

    • 由于 ArrayList 的底层实现是数组,因此迭代器的性能相对较好;
    • 而 LinkedList 的底层实现是链表,因此迭代器需要遍历链表,性能较差。

总结

ArrayList 和 LinkedList 都是线性表数据结构实现,但底层数据结构、随机访问和插入/删除操作的性能、内存占用和迭代器的性能等方面有所不同。在实际使用中,应根据具体需求选择合适的线性表数据结构实现。

我们在项目中一般是不会使用到 LinkedList 的,需要用到 LinkedList 的场景几乎都可以使用 ArrayList 来代替,并且,性能通常会更好!

# 具体实现参考

这两种数据结构的代码实现可阅读以下两篇文章:

  • 【恋上数据结构】动态数组学习笔记 (opens new window)

  • 【恋上数据结构】链表学习笔记 (opens new window)

# 什么是迭代器

迭代器是一种 Java 中的接口,用于遍历集合类(Collection)和映射类(Map)中的元素。

使用迭代器可以依次访问集合中的每个元素,而不需要知道集合的内部实现方式。

在迭代器设计模式中,迭代器提供了一个通用的访问方法,使得可以在不暴露集合内部实现的情况下对集合进行迭代遍历。

迭代器通常包含以下方法:

  • hasNext():判断集合中是否还有下一个元素,如果有返回 true,否则返回 false。
  • next():获取集合中的下一个元素。
  • remove():从集合中删除上一次返回的元素。

需要注意的是,在多线程环境下,使用迭代器遍历集合时需要进行同步操作,以避免并发修改集合导致的数据竞争和不一致性问题。

import java.util.ArrayList;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
public class ArrayListExample {
    public static void main(String[] args) {
        // 创建一个 ArrayList 对象
        List<String> arrayList = new ArrayList<>();
        // 向 ArrayList 中添加元素
        arrayList.add("apple");
        arrayList.add("banana");
        arrayList.add("orange");
        // 创建一个同步的 ArrayList 对象
        List<String> synchronizedArrayList = Collections.synchronizedList(arrayList);
        // 使用迭代器遍历 ArrayList
        synchronized (synchronizedArrayList) {
            Iterator<String> iterator = synchronizedArrayList.iterator();
            while (iterator.hasNext()) {
                String element = iterator.next();
                System.out.println(element);
            }
        }
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
上次更新: 2024/12/7 02:33:56
Java集合概述
Set接口

← Java集合概述 Set接口→

Theme by Vdoing | Copyright © 2024-2025 泪伤荡 | MIT License
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式