当前位置: 首页 > article >正文

初识数据结构——Java集合框架解析:List与ArrayList的完美结合

📚 Java集合框架解析:List与ArrayList的完美结合

🌟 前言:为什么我们需要List和ArrayList?

在日常开发中,我们经常需要处理一组数据。想象一下,如果你要管理一个班级的学生名单,或者处理电商网站的商品列表,你会怎么做?Java集合框架中的List和ArrayList就是为解决这类问题而生的利器!

本文将带你深入探索List接口和ArrayList实现类的奥秘,通过丰富的示例和直观的图示,让你彻底掌握它们的用法和原理。


🧩 第一部分:List接口全景图

1. 什么是List?

List是Java集合框架中的一个接口,它继承自Collection接口,代表一个有序的、可重复的元素序列。

public interface List<E> extends Collection<E> {// 一系列方法定义
}
🎯 List的核心特性:
  • 有序性:元素按照插入顺序排列
  • 可重复:允许存储相同的元素
  • 索引访问:可以通过下标直接访问元素

2. List的继承体系(思维导图)

Iterable
Collection
List
ArrayList
LinkedList
Vector

3. List常用方法速查表

方法签名功能描述时间复杂度
boolean add(E e)尾部添加元素O(1)
void add(int index, E element)指定位置插入O(n)
E get(int index)获取指定位置元素O(1)
E set(int index, E element)修改指定位置元素O(1)
E remove(int index)删除指定位置元素O(n)
int size()返回元素个数O(1)
boolean contains(Object o)判断是否包含元素O(n)

🚀 第二部分:ArrayList深度剖析

1. ArrayList的底层原理

ArrayList是基于动态数组实现的顺序表,它自动处理扩容问题,让我们可以专注于业务逻辑。

// 底层核心数组
transient Object[] elementData;
// 实际元素数量
private int size;

2. ArrayList的构造方法对比

构造方法说明初始容量
ArrayList()无参构造10
ArrayList(int initialCapacity)指定初始容量自定义
ArrayList(Collection<? extends E> c)从集合构造集合大小

3. 动态扩容机制揭秘(流程图)

添加元素
容量是否足够?
直接添加
计算新容量
创建新数组
拷贝数据
添加新元素

扩容规则

  • 首次添加元素时扩容到10
  • 后续按1.5倍增长(int newCapacity = oldCapacity + (oldCapacity >> 1)

4. ArrayList的三种遍历方式

List<String> list = new ArrayList<>(Arrays.asList("A", "B", "C"));// 1. for循环+下标
for (int i = 0; i < list.size(); i++) {System.out.println(list.get(i));
}// 2. 增强for循环
for (String s : list) {System.out.println(s);
}// 3. 迭代器
Iterator<String> it = list.iterator();
while (it.hasNext()) {System.out.println(it.next());
}

实战应用:扑克牌游戏

🃏1. 扑克牌游戏:买牌、洗牌、发牌完整实现

// 定义一个Card类来表示一张扑克牌
public class Card {// 定义一个整数类型的变量rank,用于表示牌的面值public int rank; // 定义一个字符串类型的变量suit,用于表示牌的花色public String suit; // 重写toString方法,用于将Card对象以特定格式输出@Overridepublic String toString() {// 格式化输出牌的花色和面值return String.format("[%s %d]", suit, rank); }
}// 导入List接口,用于存储和操作元素列表
import java.util.List; 
// 导入ArrayList类,用于创建动态数组
import java.util.ArrayList; 
// 导入Random类,用于生成随机数
import java.util.Random; // 定义一个CardDemo类,用于演示扑克牌的操作
public class CardDemo {// 定义一个字符串数组SUITS,包含四种花色public static final String[] SUITS = {"♠", "♥", "♣", "♦"}; // 定义一个静态方法buyDeck,用于创建一副完整的扑克牌private static List<Card> buyDeck() {// 创建一个容量为52的ArrayList对象,用于存储扑克牌List<Card> deck = new ArrayList<>(52); // 外层循环遍历四种花色for (int i = 0; i < 4; i++) { // 内层循环遍历1到13的牌面值for (int j = 1; j <= 13; j++) { // 获取当前花色String suit = SUITS[i]; // 获取当前牌面值int rank = j; // 创建一个Card对象Card card = new Card(); // 设置Card对象的牌面值card.rank = rank; // 设置Card对象的花色card.suit = suit; // 将Card对象添加到deck列表中deck.add(card); }}// 返回包含所有扑克牌的列表return deck; }// 定义一个静态方法swap,用于交换列表中两个位置的元素private static void swap(List<Card> deck, int i, int j) {// 获取索引i位置的Card对象Card t = deck.get(i); // 将索引j位置的Card对象赋值给索引i位置deck.set(i, deck.get(j)); // 将临时变量t(原索引i位置的Card对象)赋值给索引j位置deck.set(j, t); }// 定义一个静态方法shuffle,用于洗牌private static void shuffle(List<Card> deck) {// 创建一个Random对象,使用固定的种子值,保证每次运行结果一致Random random = new Random(20190905); // 从列表的最后一个元素开始向前遍历for (int i = deck.size() - 1; i > 0; i--) { // 生成一个0到i之间的随机整数int r = random.nextInt(i); // 调用swap方法交换索引i和r位置的元素swap(deck, i, r); }}// 程序的入口点public static void main(String[] args) {// 调用buyDeck方法创建一副扑克牌List<Card> deck = buyDeck(); // 打印提示信息,表示刚买回来的牌System.out.println("刚买回来的牌:"); // 打印刚买回来的扑克牌列表System.out.println(deck); // 调用shuffle方法对扑克牌进行洗牌shuffle(deck); // 打印提示信息,表示洗过的牌System.out.println("洗过的牌:"); // 打印洗过的扑克牌列表System.out.println(deck); // 创建一个二维列表hands,用于存储三个玩家的手牌List<List<Card>> hands = new ArrayList<>(); // 为每个玩家创建一个空的手牌列表,并添加到hands中hands.add(new ArrayList<>()); hands.add(new ArrayList<>()); hands.add(new ArrayList<>()); // 模拟每个玩家轮流抓5张牌的过程for (int i = 0; i < 5; i++) { // 遍历三个玩家for (int j = 0; j < 3; j++) { // 从deck列表中移除第一张牌,并添加到当前玩家的手牌列表中hands.get(j).add(deck.remove(0)); }}// 打印提示信息,表示剩余的牌System.out.println("剩余的牌:"); // 打印剩余的扑克牌列表System.out.println(deck); // 打印提示信息,表示A玩家手中的牌System.out.println("A手中的牌:"); // 打印A玩家的手牌列表System.out.println(hands.get(0)); // 打印提示信息,表示B玩家手中的牌System.out.println("B手中的牌:"); // 打印B玩家的手牌列表System.out.println(hands.get(1)); // 打印提示信息,表示C玩家手中的牌System.out.println("C手中的牌:"); // 打印C玩家的手牌列表System.out.println(hands.get(2)); }
}

🔺2. 杨辉三角生成器(leetcode118)

在这里插入图片描述

public List<List<Integer>> generate(int numRows) {List<List<Integer>> triangle = new ArrayList<>();for (int i = 0; i < numRows; i++) {List<Integer> row = new ArrayList<>();for (int j = 0; j <= i; j++) {if (j == 0 || j == i) {row.add(1);} else {row.add(triangle.get(i-1).get(j-1) + triangle.get(i-1).get(j));}}triangle.add(row);}return triangle;}

💡 性能优化与思考

ArrayList的优缺点分析

优点

  • 随机访问速度快(O(1))
  • 内存连续,缓存友好
  • 尾部操作高效

缺点

  • 中间插入/删除效率低(O(n))
  • 扩容有性能开销
  • 可能造成内存浪费

替代方案考虑

场景推荐结构原因
频繁随机访问ArrayListO(1)访问
频繁插入删除LinkedListO(1)插入删除
多线程环境CopyOnWriteArrayList线程安全
固定大小Arrays.asList()不可变

📌 总结与面试必备

  1. List vs ArrayList:List是接口,ArrayList是实现
  2. 扩容机制:初始10,1.5倍增长
  3. 时间复杂度
    • 访问:O(1)
    • 搜索:O(n)
    • 插入/删除:平均O(n)
  4. 线程安全:ArrayList非线程安全,多线程需同步
45% 30% 15% 10% ArrayList使用场景 随机访问 尾部操作 中间操作 其他

🎁 彩蛋:ArrayList的趣味事实

你知道吗?ArrayList在JDK1.2中引入,它的设计受到了C++ STL中vector的启发。但在Java中,为了避免与数学向量概念混淆,才命名为ArrayList!


希望这篇深度解析能帮助你彻底掌握List和ArrayList!如果有任何问题,欢迎在评论区留言讨论。别忘了点赞收藏哦~ 💖

相关文章:

初识数据结构——Java集合框架解析:List与ArrayList的完美结合

&#x1f4da; Java集合框架解析&#xff1a;List与ArrayList的完美结合 &#x1f31f; 前言&#xff1a;为什么我们需要List和ArrayList&#xff1f; 在日常开发中&#xff0c;我们经常需要处理一组数据。想象一下&#xff0c;如果你要管理一个班级的学生名单&#xff0c;或…...

TDengine 从入门到精通(2万字长文)

目录 第一章:走进 TDengine 的世界 TDengine 是个啥? TDengine 的硬核特性 性能炸裂 分布式架构,天生可扩展 SQL 用起来贼顺手 写入方式花样多 内置缓存,省心又省力 TDengine 能干啥? 智能制造 能源管理 物联网平台 工业大数据 第二章:上手 TDengine:安装与…...

DevOps 与持续集成(CI/CD)

1. DevOps 概述 DevOps(Development + Operations)是一种软件开发方法,强调开发(Dev)与运维(Ops)协作,通过自动化工具提高软件交付效率。其目标是: ✅ 提高部署速度 —— 频繁发布新版本 ✅ 减少人为错误 —— 通过自动化降低运维风险 ✅ 增强可观测性 —— 监控和日…...

[特殊字符] 使用 Handsontable 构建一个支持 Excel 公式计算的动态表格

在 Web 应用中&#xff0c;处理表格数据并提供 Excel 级的功能&#xff08;如公式计算、数据导入导出&#xff09;一直是个挑战。今天&#xff0c;我将带你使用 React Handsontable 搭建一个强大的 Excel 风格表格&#xff0c;支持 公式计算、Excel 文件导入导出&#xff0c;并…...

uniapp微信小程序引入vant组件库

1、首先要有uniapp项目&#xff0c;根据vant官方文档使用yarn或npm安装依赖&#xff1a; 1、 yarn init 或 npm init2、 # 通过 npm 安装npm i vant/weapp -S --production# 通过 yarn 安装yarn add vant/weapp --production# 安装 0.x 版本npm i vant-weapp -S --production …...

贪心进阶学习笔记

反悔贪心 贪心是指直接选择局部最优解&#xff0c;不需要考虑之后的影响。 而反悔贪心是在贪心上面加了一个“反悔”的操作&#xff0c;于是又可以撤销之前的“鲁莽行动”&#xff0c;让整个的选择稍微变得“理智一些”。 于是&#xff0c;我个人理解&#xff0c;反悔贪心是…...

Java 大视界 -- Java 大数据在航天遥测数据分析中的技术突破与应用(177)

&#x1f496;亲爱的朋友们&#xff0c;热烈欢迎来到 青云交的博客&#xff01;能与诸位在此相逢&#xff0c;我倍感荣幸。在这飞速更迭的时代&#xff0c;我们都渴望一方心灵净土&#xff0c;而 我的博客 正是这样温暖的所在。这里为你呈上趣味与实用兼具的知识&#xff0c;也…...

架构师面试(二十七):单链表

问题 今天的问题对于架构师来说会相对容易许多。今天出一个【数据结构与算法】相关的题目&#xff0c;醒醒脑。 给一张【单链表】&#xff0c;该单链表有100个节点元素&#xff08;当然&#xff0c;事先我们是不知道100这个数目的&#xff09;&#xff0c;要获取倒数第8个元素…...

从扩展黎曼泽塔函数构造物质和时空的结构-15

回来考虑泽塔函数&#xff0c; 我们知道&#xff0c; 也就是在平面直角坐标系上反正切函数在x上的变化率&#xff0c;那么不难看出&#xff0c; 就是在s维空间上的“广义”反正切函数在单位p上的变化率&#xff0c;而泽塔函数&#xff0c;就是这些变化率的全乘积&#xff0c; 因…...

数据库访问工具 dbVisitor v6.0.0 发布

dbVisitor 是一款轻量小巧、功能完备的 Java 数据库 ORM 工具&#xff0c;它的前身是 HasorDB&#xff0c;历经 8 年迭代后正式更名为 dbVisitor 并开始独立发展4。以下是关于 dbVisitor v6.0.0 发布的相关信息&#xff1a; 发布说明 在 Maven Central 上可查询到 dbVisitor …...

01背包问题详解 具体样例模拟版

01背包 有 N 件物品和一个容量是 V 的背包。每件物品只能使用一次。 第 i 件物品的体积是 v i v_i vi​&#xff0c;价值是 w i w_i wi​。 求解将哪些物品装入背包&#xff0c;可使这些物品的总体积不超过背包容量&#xff0c;且总价值最大。 输出最大价值。 输入格式 …...

网络初识 - Java

网络发展史&#xff1a; 单机时代&#xff08;独立模式&#xff09; -> 局域网时代 -> 广域网时代 -> 移动互联网时代 网络互联&#xff1a;将多台计算机链接再一起&#xff0c;完成数据共享。 数据共享的本质是网络数据传输&#xff0c;即计算机之间通过网络来传输数…...

zk基础—5.Curator的使用与剖析一

大纲 1.基于Curator进行基本的zk数据操作 2.基于Curator实现集群元数据管理 3.基于Curator实现HA主备自动切换 4.基于Curator实现Leader选举 5.基于Curator实现分布式Barrier 6.基于Curator实现分布式计数器 7.基于Curator实现zk的节点和子节点监听机制 8.基于Curator创…...

大模型快速 ASGI 服务器uvicorn

基础概念类 1. 什么是 Uvicorn&#xff0c;它的作用是什么&#xff1f; 答案&#xff1a;Uvicorn 是一个基于 Python 的快速 ASGI&#xff08;异步服务器网关接口&#xff09;服务器。它的主要作用是作为 Web 应用程序的服务器&#xff0c;负责接收客户端的请求&#xff0c;并…...

每日一题(小白)回溯篇4

深度优先搜索题&#xff1a;找到最长的路径&#xff0c;计算这样的路径有多少条&#xff08;使用回溯&#xff09; 分析题意可以得知&#xff0c;每次向前后左右走一步&#xff0c;直至走完16步就算一条走通路径。要求条件是不能超出4*4的范围&#xff0c;不能重复之前的路径。…...

消息队列基础概念及选型,常见解决方案包括消息可靠性、消息有序、消息堆积、重复消费、事务消息

前言 是时候总结下消息队列相关知识点啦&#xff01;我搓搓搓搓 本文包括消息队列基础概念介绍&#xff0c;常见解决方案包括消息可靠性、消息有序、消息堆积、重复消费、事务消息 参考资料&#xff1a; Kafka常见问题总结 | JavaGuide RocketMQ常见问题总结 | JavaGuide …...

基于STM32与应变片的协作机械臂力反馈控制系统设计与实现---3.3 机械结构改装

3.3 机械臂结构改装设计与实施 一、改装需求分析 1.1 改装类型分级 改装级别涉及范围典型改动周期成本I级(小型)末端执行器工具快换装置1-3天$500-2000II级(中型)关节模块电机/减速器升级1-2周$2000-8000III级(大型)本体结构材质/拓扑优化1-3月$8000+1.2 关键参数变更评…...

k8s进阶之路:本地集群环境搭建

概述 文章将带领大家搭建一个 master 节点&#xff0c;两个 node 节点的 k8s 集群&#xff0c;容器基于 docker&#xff0c;k8s 版本 v1.32。 一、系统安装 安装之前请大家使用虚拟机将 ubuntu24.04 系统安装完毕&#xff0c;我是基于 mac m1 的系统进行安装的&#xff0c;所…...

云服务器实战:用 Nginx 搭建高性能 API 网关与反向代理服务(附完整配置流程)

在如今的 Web 系统架构中&#xff0c;“接口统一出口”已成为必备设计模式——无论是前后端分离、微服务架构&#xff0c;还是多端接入&#xff08;Web、小程序、App&#xff09;&#xff0c;一个稳定、高性能、可扩展的 API 网关至关重要。 而 Nginx&#xff0c;作为轻量级高…...

C++ STL 详解 ——list 的深度解析与实践指南

在 C 的标准模板库&#xff08;STL&#xff09;中&#xff0c;list作为一种重要的序列式容器&#xff0c;以其独特的双向链表结构和丰富的操作功能&#xff0c;在许多编程场景下发挥着关键作用。深入理解list的特性与使用方法&#xff0c;能帮助开发者编写出更高效、灵活的代码…...

按键切换LCD显示后,显示总在第二阶段,而不在第一阶段的问题

这是一个密码锁的程序&#xff0c;当在输入密码后&#xff0c;原本是要重置密码&#xff0c;但是程序总是在输入密码正确后总是跳转置设置第二个密码&#xff0c;而第一个密码总是跳过。 不断修改后&#xff0c; 解决方法 将if语句换成switch语句&#xff0c;这样就可以分离程序…...

护网蓝初面试题

《网安面试指南》https://mp.weixin.qq.com/s/RIVYDmxI9g_TgGrpbdDKtA?token1860256701&langzh_CN 5000篇网安资料库https://mp.weixin.qq.com/s?__bizMzkwNjY1Mzc0Nw&mid2247486065&idx2&snb30ade8200e842743339d428f414475e&chksmc0e4732df793fa3bf39…...

C++11: 智能指针

C11: 智能指针 &#xff08;一&#xff09;智能指针原理1.RAll2.智能指针 (二)C11 智能指针1. auto_ptr2. unique_ptr3. shared_ptr4. weak_ptr &#xff08;三&#xff09;shared_ptr中存在的问题std::shared_ptr的循环引用 &#xff08;四&#xff09;删除器&#xff08;五&a…...

去中心化预测市场

去中心化预测市场 核心概念 预测市场类型&#xff1a; 类别型市场&#xff1a;二元结果&#xff08;YES/NO&#xff09;&#xff0c;例如“BTC在2024年突破10万美元&#xff1f;” 多选型市场&#xff1a;多个选项&#xff08;如总统候选人&#xff09;&#xff0c;赔付基于…...

从零实现本地大模型RAG部署

1. RAG概念 RAG&#xff08;Retrieval-Augmented Generation&#xff09;即检索增强生成&#xff0c;是一种结合信息检索与大型语言模型&#xff08;大模型&#xff09;的技术。从外部知识库&#xff08;如文档、数据库或网页&#xff09;中实时检索相关信息&#xff0c;并将其…...

使用 Python 连接 PostgreSQL 数据库,从 `mimic - III` 数据库中筛选数据并导出特定的数据图表

要使用 Python 连接 PostgreSQL 数据库&#xff0c;从 mimic - III 数据库中筛选数据并导出特定的数据图表&#xff0c;你可以按照以下步骤操作&#xff1a; 安装所需的库&#xff1a;psycopg2 用于连接 PostgreSQL 数据库&#xff0c;pandas 用于数据处理&#xff0c;matplot…...

【Linux系统篇】:探索文件系统原理--硬件磁盘、文件系统与链接的“三体宇宙”

✨感谢您阅读本篇文章&#xff0c;文章内容是个人学习笔记的整理&#xff0c;如果哪里有误的话还请您指正噢✨ ✨ 个人主页&#xff1a;余辉zmh–CSDN博客 ✨ 文章所属专栏&#xff1a;Linux篇–CSDN博客 文章目录 一.认识硬件--磁盘物理存储结构1.存储介质类型2.物理存储单元3…...

Tracing the thoughts of a large language model 简单理解

Tracing the thoughts of a large language model 这篇论文通过电路追踪方法(Circuit Tracing)揭示了大型语言模型Claude 3.5 Haiku的内部机制,其核心原理可归纳为以下几个方面: 1. 方法论核心:归因图与替换模型 替换模型(Replacement Model) 使用跨层转码器(CLT)将原…...

OpenCV边缘检测技术详解:原理、实现与应用

概述 边缘检测是计算机视觉和图像处理中最基本也是最重要的技术之一&#xff0c;它通过检测图像中亮度或颜色急剧变化的区域来识别物体的边界。边缘通常对应着场景中物体的物理边界、表面方向的变化或深度不连续处。 分类 OpenCV提供了多种边缘检测算法&#xff0c;下面我们介…...

Java学习——day22(Java反射基础入门)

文章目录 1.反射的定义2. 认识反射的关键API2.1 Class2.2 Field2.3 Method2.4 Constructor 3. 示例代码讲解与分析4. 编写反射示例代码的步骤4.1 定义测试类4.2 编写主程序&#xff0c;使用反射获取信息4.3 通过反射创建对象并调用方法 5. 总结6.今日生词 Java反射笔记 1.反射的…...