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

news2025/4/9 15:36:58

📚 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对象以特定格式输出
    @Override
    public 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!如果有任何问题,欢迎在评论区留言讨论。别忘了点赞收藏哦~ 💖

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

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

相关文章

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; 因…

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;即计算机之间通过网络来传输数…

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

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

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

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

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

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

【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;下面我们介…

BN 层做预测的时候, 方差均值怎么算

✅ 一、Batch Normalization&#xff08;BN&#xff09;回顾 BN 层在训练和推理阶段的行为是不一样的&#xff0c;核心区别就在于&#xff1a; 训练时用 mini-batch 里的均值方差&#xff0c;预测时用全局的“滑动平均”均值方差。 &#x1f9ea; 二、训练阶段&#xff08;Trai…

JS 其他事件类型

页面加载 事件 window.addEvent() window.addEventListener(load,function(){const btn document.querySelector(button)btn.addEventListener(click,function(){alert(按钮)})})也可以给其他标签加该事件 HTML加载事件 找html标签 也可以给页面直接赋值

AI Agent设计模式五:Orchestrator

概念 &#xff1a;中央任务调度中枢 ✅ 优点&#xff1a;全局资源协调&#xff0c;确保任务执行顺序❌ 缺点&#xff1a;单点故障风险&#xff0c;可能成为性能瓶颈 import operator import osfrom langchain.schema import SystemMessage, HumanMessage from langchain_opena…