初识数据结构——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的继承体系(思维导图)
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))
- 扩容有性能开销
- 可能造成内存浪费
替代方案考虑
| 场景 | 推荐结构 | 原因 |
|---|---|---|
| 频繁随机访问 | ArrayList | O(1)访问 |
| 频繁插入删除 | LinkedList | O(1)插入删除 |
| 多线程环境 | CopyOnWriteArrayList | 线程安全 |
| 固定大小 | Arrays.asList() | 不可变 |
📌 总结与面试必备
- List vs ArrayList:List是接口,ArrayList是实现
- 扩容机制:初始10,1.5倍增长
- 时间复杂度:
- 访问:O(1)
- 搜索:O(n)
- 插入/删除:平均O(n)
- 线程安全:ArrayList非线程安全,多线程需同步
🎁 彩蛋:ArrayList的趣味事实
你知道吗?ArrayList在JDK1.2中引入,它的设计受到了C++ STL中vector的启发。但在Java中,为了避免与数学向量概念混淆,才命名为ArrayList!
希望这篇深度解析能帮助你彻底掌握List和ArrayList!如果有任何问题,欢迎在评论区留言讨论。别忘了点赞收藏哦~ 💖
相关文章:
初识数据结构——Java集合框架解析:List与ArrayList的完美结合
📚 Java集合框架解析:List与ArrayList的完美结合 🌟 前言:为什么我们需要List和ArrayList? 在日常开发中,我们经常需要处理一组数据。想象一下,如果你要管理一个班级的学生名单,或…...
TDengine 从入门到精通(2万字长文)
目录 第一章:走进 TDengine 的世界 TDengine 是个啥? TDengine 的硬核特性 性能炸裂 分布式架构,天生可扩展 SQL 用起来贼顺手 写入方式花样多 内置缓存,省心又省力 TDengine 能干啥? 智能制造 能源管理 物联网平台 工业大数据 第二章:上手 TDengine:安装与…...
DevOps 与持续集成(CI/CD)
1. DevOps 概述 DevOps(Development + Operations)是一种软件开发方法,强调开发(Dev)与运维(Ops)协作,通过自动化工具提高软件交付效率。其目标是: ✅ 提高部署速度 —— 频繁发布新版本 ✅ 减少人为错误 —— 通过自动化降低运维风险 ✅ 增强可观测性 —— 监控和日…...
[特殊字符] 使用 Handsontable 构建一个支持 Excel 公式计算的动态表格
在 Web 应用中,处理表格数据并提供 Excel 级的功能(如公式计算、数据导入导出)一直是个挑战。今天,我将带你使用 React Handsontable 搭建一个强大的 Excel 风格表格,支持 公式计算、Excel 文件导入导出,并…...
uniapp微信小程序引入vant组件库
1、首先要有uniapp项目,根据vant官方文档使用yarn或npm安装依赖: 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 …...
贪心进阶学习笔记
反悔贪心 贪心是指直接选择局部最优解,不需要考虑之后的影响。 而反悔贪心是在贪心上面加了一个“反悔”的操作,于是又可以撤销之前的“鲁莽行动”,让整个的选择稍微变得“理智一些”。 于是,我个人理解,反悔贪心是…...
Java 大视界 -- Java 大数据在航天遥测数据分析中的技术突破与应用(177)
💖亲爱的朋友们,热烈欢迎来到 青云交的博客!能与诸位在此相逢,我倍感荣幸。在这飞速更迭的时代,我们都渴望一方心灵净土,而 我的博客 正是这样温暖的所在。这里为你呈上趣味与实用兼具的知识,也…...
架构师面试(二十七):单链表
问题 今天的问题对于架构师来说会相对容易许多。今天出一个【数据结构与算法】相关的题目,醒醒脑。 给一张【单链表】,该单链表有100个节点元素(当然,事先我们是不知道100这个数目的),要获取倒数第8个元素…...
从扩展黎曼泽塔函数构造物质和时空的结构-15
回来考虑泽塔函数, 我们知道, 也就是在平面直角坐标系上反正切函数在x上的变化率,那么不难看出, 就是在s维空间上的“广义”反正切函数在单位p上的变化率,而泽塔函数,就是这些变化率的全乘积, 因…...
数据库访问工具 dbVisitor v6.0.0 发布
dbVisitor 是一款轻量小巧、功能完备的 Java 数据库 ORM 工具,它的前身是 HasorDB,历经 8 年迭代后正式更名为 dbVisitor 并开始独立发展4。以下是关于 dbVisitor v6.0.0 发布的相关信息: 发布说明 在 Maven Central 上可查询到 dbVisitor …...
01背包问题详解 具体样例模拟版
01背包 有 N 件物品和一个容量是 V 的背包。每件物品只能使用一次。 第 i 件物品的体积是 v i v_i vi,价值是 w i w_i wi。 求解将哪些物品装入背包,可使这些物品的总体积不超过背包容量,且总价值最大。 输出最大价值。 输入格式 …...
网络初识 - Java
网络发展史: 单机时代(独立模式) -> 局域网时代 -> 广域网时代 -> 移动互联网时代 网络互联:将多台计算机链接再一起,完成数据共享。 数据共享的本质是网络数据传输,即计算机之间通过网络来传输数…...
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,它的作用是什么? 答案:Uvicorn 是一个基于 Python 的快速 ASGI(异步服务器网关接口)服务器。它的主要作用是作为 Web 应用程序的服务器,负责接收客户端的请求,并…...
每日一题(小白)回溯篇4
深度优先搜索题:找到最长的路径,计算这样的路径有多少条(使用回溯) 分析题意可以得知,每次向前后左右走一步,直至走完16步就算一条走通路径。要求条件是不能超出4*4的范围,不能重复之前的路径。…...
消息队列基础概念及选型,常见解决方案包括消息可靠性、消息有序、消息堆积、重复消费、事务消息
前言 是时候总结下消息队列相关知识点啦!我搓搓搓搓 本文包括消息队列基础概念介绍,常见解决方案包括消息可靠性、消息有序、消息堆积、重复消费、事务消息 参考资料: 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 节点,两个 node 节点的 k8s 集群,容器基于 docker,k8s 版本 v1.32。 一、系统安装 安装之前请大家使用虚拟机将 ubuntu24.04 系统安装完毕,我是基于 mac m1 的系统进行安装的,所…...
云服务器实战:用 Nginx 搭建高性能 API 网关与反向代理服务(附完整配置流程)
在如今的 Web 系统架构中,“接口统一出口”已成为必备设计模式——无论是前后端分离、微服务架构,还是多端接入(Web、小程序、App),一个稳定、高性能、可扩展的 API 网关至关重要。 而 Nginx,作为轻量级高…...
C++ STL 详解 ——list 的深度解析与实践指南
在 C 的标准模板库(STL)中,list作为一种重要的序列式容器,以其独特的双向链表结构和丰富的操作功能,在许多编程场景下发挥着关键作用。深入理解list的特性与使用方法,能帮助开发者编写出更高效、灵活的代码…...
按键切换LCD显示后,显示总在第二阶段,而不在第一阶段的问题
这是一个密码锁的程序,当在输入密码后,原本是要重置密码,但是程序总是在输入密码正确后总是跳转置设置第二个密码,而第一个密码总是跳过。 不断修改后, 解决方法 将if语句换成switch语句,这样就可以分离程序…...
护网蓝初面试题
《网安面试指南》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: 智能指针 (一)智能指针原理1.RAll2.智能指针 (二)C11 智能指针1. auto_ptr2. unique_ptr3. shared_ptr4. weak_ptr (三)shared_ptr中存在的问题std::shared_ptr的循环引用 (四)删除器(五&a…...
去中心化预测市场
去中心化预测市场 核心概念 预测市场类型: 类别型市场:二元结果(YES/NO),例如“BTC在2024年突破10万美元?” 多选型市场:多个选项(如总统候选人),赔付基于…...
从零实现本地大模型RAG部署
1. RAG概念 RAG(Retrieval-Augmented Generation)即检索增强生成,是一种结合信息检索与大型语言模型(大模型)的技术。从外部知识库(如文档、数据库或网页)中实时检索相关信息,并将其…...
使用 Python 连接 PostgreSQL 数据库,从 `mimic - III` 数据库中筛选数据并导出特定的数据图表
要使用 Python 连接 PostgreSQL 数据库,从 mimic - III 数据库中筛选数据并导出特定的数据图表,你可以按照以下步骤操作: 安装所需的库:psycopg2 用于连接 PostgreSQL 数据库,pandas 用于数据处理,matplot…...
【Linux系统篇】:探索文件系统原理--硬件磁盘、文件系统与链接的“三体宇宙”
✨感谢您阅读本篇文章,文章内容是个人学习笔记的整理,如果哪里有误的话还请您指正噢✨ ✨ 个人主页:余辉zmh–CSDN博客 ✨ 文章所属专栏: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边缘检测技术详解:原理、实现与应用
概述 边缘检测是计算机视觉和图像处理中最基本也是最重要的技术之一,它通过检测图像中亮度或颜色急剧变化的区域来识别物体的边界。边缘通常对应着场景中物体的物理边界、表面方向的变化或深度不连续处。 分类 OpenCV提供了多种边缘检测算法,下面我们介…...
Java学习——day22(Java反射基础入门)
文章目录 1.反射的定义2. 认识反射的关键API2.1 Class2.2 Field2.3 Method2.4 Constructor 3. 示例代码讲解与分析4. 编写反射示例代码的步骤4.1 定义测试类4.2 编写主程序,使用反射获取信息4.3 通过反射创建对象并调用方法 5. 总结6.今日生词 Java反射笔记 1.反射的…...
