黑马Java面试笔记之 集合篇(算法复杂度+ArrayList+)

news2025/6/6 1:26:21

一. 算法复杂度分析

1.1 时间复杂度

        时间复杂度分析:来评估代码的执行耗时的

常见的复杂度表示形式

常见复杂度

1.2 空间复杂度

        空间复杂度全称是渐进空间复杂度,表示算法占用的额外存储空间数据规模之间的增长关系

二. 数组

        数组(Array)是一种用连续的内存空间存储相同数据类型数据的线性数据结构

int[] array = {22,33,88,66,55,25};

我们定义了这么一个数组之后,在内存的表示是这样的:

        现在假如,我们通过arrar[1],想要获得下标为1这个元素,但是现在栈内存中指向的堆内存数组的首地址,它是如何获取下标为1这个数据的?

2.1 寻址公式

在数组在内存中查找元素的时候,是有一个寻址公式的,如下:

arr[i] = baseAddress + i * dataTypeSize

baseAddress:数组的首地址,目前是10

dataTypeSize:代表数组中元素类型的大小,目前数组重存储的是int型的数据,dataTypeSize=4个字节

arr:指的是数组

i:指的是数组的下标

有了寻址公式以后,我们再来获取一下下标为1的元素,这个是原来的数组

int[] array = {22,33,88,66,55,25};

套入公式:

array[1] =10 + i * 4 = 14

获取到14这个地址,就能获取到下标为1的这个元素了。

2.2 操作数组的时间复杂度(查找)

        随机查询(根据索引查询)

数组元素的访问是通过下标来访问的,计算机通过数组的首地址和寻址公式能够很快速的找到想要访问的元素

        未知索引查询

情况一:查找数组内的元素,查找55号数据

情况二:查找排序后数组内的元素,查找55号数据

2.3 操作数组的时间复杂度(插入、删除)

        数组是一段连续的内存空间,因此为了保证数组的连续性会使得数组的插入和删除的效率变得很低

总结

三. ArrayList源码分析

分析ArrayList源码主要从三个方面去翻阅:成员变量,构造函数,关键方法

3.1 成员变量

DEFAULT_CAPACITY = 10; 默认初始的容量**(CAPACITY)

EMPTY_ELEMENTDATA = {}; 用于空实例的共享空数组实例

DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};用于默认大小的空实例的共享空数组实例

Object[] elementData; 存储元素的数组缓冲区

int size; ArrayList的大小(它包含的元素数量)

3.2 构造方法

  • 第一个构造是带初始化容量的构造函数,可以按照指定的容量初始化数组

  • 第二个是无参构造函数,默认创建一个空集合

将collection对象转换成数组,然后将数组的地址的赋给elementData

3.3 ArrayList源码分析

添加数据的流程

    四.  ArrayList底层的实现原理

    结论:

    • ArrayList底层是用动态的数组 实现的
    • ArrayList初始容量为0,当第一次添加数据的时候才会初始化容量为10
    • ArrayList在进行扩容的时候是原来容量的1.5倍,每次扩容都需要拷贝数组
    • ArrayList在添加数据的时候
      • 确保数组已使用长度(size)加1之后足够存下下一个数据

      • 计算数组的容量,如果当前数组已使用长度+1后的大于当前的数组长度,则调用grow方法扩容(原来的1.5倍)

      • 确保新增的数据有地方存储之后,则将新元素添加到位于size的位置上。

      • 返回添加成功布尔值。

    五. ArrayList list=new ArrayList(10)中的list扩容几次

    答:该语句只是声明和实例了一个ArrayList,指定了容量为10,未扩容

    六. 数组和List之间的转换

    如何实现数组和List之间的转换?

    数组转List,使用JDK中java.util.Arrays工具类的asList方法

    List转数组,使用List的toArray方法。无参toArray方法返回 Object数组,传入初始化长度的数组对象,返回该对象数组

    用Arrays.asList转List后,如果修改了数组内容,list受影响吗

    List用toArray转数组后,如果修改了List内容,数组受影响吗

    回答:

    1.用Arrays.asList转List后,如果修改了数组内容,list受影响吗

    Arrays.asList转换list之后,如果修改了数组的内容,list会受影响,因为它的底层使用的Arrays类中的一个内部类ArrayList来构造的集合,在这个集合的构造器中,把我们传入的这个集合进行了包装而已,最终指向的都是同一个内存地址

    2,List用toArray转数组后,如果修改了List内容,数组受影响吗

    list用了toArray转数组后,如果修改了list内容,数组不会影响,当调用了toArray以后,在底层是它是进行了数组的拷贝,跟原来的元素就没啥关系了,所以即使list修改了以后,数组也不受影响

    七. LinkedList数据结构链表

    7.1 单向链表

    • 链表中的每一个元素称之为结点(Node)

    • 物理存储单元上,非连续、非顺序的存储结构

    • 单向链表:每个结点包括两个部分:一个是存储数据元素的数据域,另一个是存储下一个结点地址的指针域。记录下个结点地址的指针叫作后继指针 next

    7.2 单向链表时间复杂度分析

    • 查询操作

    • 插入/删除操作

    7.3 双向链表

    而双向链表,顾名思义,它支持两个方向

    • 每个结点不止有一个后继指针 next 指向后面的结点

    • 有一个前驱指针 prev 指向前面的结点

    双向链表时间复杂度

    总结

    八. ArrayList和LinkedLis区别

    面试官:面试题-ArrayList和LinkedList的区别是什么?

    候选人

    • 底层数据结构

      • ArrayList 是动态数组的数据结构实现

      • LinkedList 是双向链表的数据结构实现

    • 操作数据效率

      • ArrayList按照下标查询的时间复杂度O(1)【内存是连续的,根据寻址公式】, LinkedList不支持下标查询

      • 查找(未知索引): ArrayList需要遍历,链表也需要遍历,时间复杂度都是O(n)

      • 新增和删除

        • ArrayList尾部插入和删除,时间复杂度是O(1);其他部分增删需要挪动数组,时间复杂度是O(n)

        • LinkedList头尾节点增删时间复杂度是O(1),其他都需要遍历链表,时间复杂度是O(n)

    • 内存空间占用

      • ArrayList底层是数组,内存连续,节省内存

      • LinkedList 是双向链表需要存储数据,和两个指针,更占用内存

    • 线程安全

      • ArrayList和LinkedList都不是线程安全的

      • 如果需要保证线程安全,有两种方案:

        • 在方法内使用,局部变量则是线程安全的

        • 使用线程安全的ArrayList和LinkedList

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

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

    相关文章

    Nginx+Tomcat负载均衡

    目录 Tomcat简介 Tomcat 的核心功能 Tomcat架构 Tomcat 的特点 Tomact配置 关闭防火墙及系统内核 Tomcar 主要文件信息 配置文件说明 案例一:Java的Web站点 案例二:NginxTomcat负载均衡、动静分离 Tomcat简介 Tomcat 是由 Apache 软件基金会&am…

    【论文解读】ReAct:从思考脱离行动, 到行动反馈思考

    认识从实践开始,经过实践得到了理论的认识,还须再回到实践去。 ——《实践论》,毛泽东 1st author: About – Shunyu Yao – 姚顺雨 paper [2210.03629] ReAct: Synergizing Reasoning and Acting in Language ModelsReAct: Synergizing Reasoning and…

    简单配置RHEL9.X

    切换默认运行级别 将系统默认启动模式从多用户的图形界面调整为多用户的文本界面,适用于优化系统资源占用或进行远程服务器管理的场景。 注意:安装选择“带GUI的服务器”部分常用命令默认安装;如果选择“最小安装”时,部分常用命…

    下载并运行自制RAG框架

    项目部署 https://github.com/huangjia2019/rag-project01-framework git clone https://github.com/huangjia2019/rag-project01-framework.git 一 、 前端分部分部署 在 Ubuntu 系统 上安装 Node.js 和 npm(Node Package Manager),并初始…

    Rust 学习笔记:Cargo 工作区

    Rust 学习笔记:Cargo 工作区 Rust 学习笔记:Cargo 工作区创建工作区在工作区中创建第二个包依赖于工作区中的外部包向工作区添加测试将工作区中的 crate 发布到 crates.io添加 add_two crate 到工作区总结 Rust 学习笔记:Cargo 工作区 随着项…

    颈部的 “异常坚持”

    生活中,有些人的颈部会突然变得 “异常坚持”—— 头部不受控制地偏向一侧,或是不自主地旋转、后仰,仿佛被无形的力量牵引着。这种情况不仅影响外观,还会带来强烈的不适感,颈部肌肉紧绷、酸痛,像被一根绳索…

    Ubuntu22.04安装MinkowskiEngine

    MinkowskiEngine简介 Minkowski引擎是一个用于稀疏张量的自动微分库。它支持所有标准神经网络层,例如对稀疏张量的卷积、池化和广播操作。 MinkowskiEngine安装 官方源码链接:GitHub - NVIDIA/MinkowskiEngine: Minkowski Engine is an auto-diff neu…

    【计算机网络】第2章:应用层—应用层协议原理

    目录 1. 网络应用的体系结构 2. 客户-服务器(C/S)体系结构 3. 对等体(P2P)体系结构 4. C/S 和 P2P 体系结构的混合体 Napster 即时通信 5. 进程通信 6. 分布式进程通信需要解决的问题 7. 问题1:对进程进行编址…

    【Zephyr 系列 6】使用 Zephyr + BLE 打造蓝牙广播与连接系统(STEVAL-IDB011V1 实战)

    🧠关键词:Zephyr、BLE、广播、连接、GATT、低功耗蓝牙、STEVAL-IDB011V1 📌适合人群:希望基于 Zephyr 实现 BLE 通信的嵌入式工程师、蓝牙产品开发人员 🧭 前言:为什么选择 Zephyr 开发 BLE? 在传统 BLE 开发中,我们大多依赖于厂商 SDK(如 Nordic SDK、BlueNRG SD…

    利用 Scrapy 构建高效网页爬虫:框架解析与实战流程

    目录 前言1 Scrapy 框架概述1.1 Scrapy 的核心优势1.2 Scrapy 的典型应用场景 2 Scrapy 工作原理解析2.1 框架结构图2.2 Spider:定义数据采集策略2.3 Scheduler:调度请求与去重2.4 Downloader:网页下载器2.5 Item:结构化数据容器2…

    RPG20.创建敌人的初始能力和加载武器

    1. 基于StartUpAbilitiy基类创建专门用于敌人数据的DAStartUpABility,然后再基于新创建的DA再创建一个蓝图 2.打开 DataAsset_EnemyStartUpData.h // Fill out your copyright notice in the Description page of Project Settings.#pragma once#include "Cor…

    自适应移动平均(Adaptive Moving Average, AMA)

    文章目录 1. 考夫曼自适应移动平均 (KAMA)算法推导及Python实现2. 对 (KAMA)算法参数进行优化及实现 自适应移动平均(Adaptive Moving Average, AMA)由Perry Kaufman在其著作《Trading Systems and Methods》中提出,它通过动态调整平滑系数来…

    涨薪技术|0到1学会性能测试第95课-全链路脚本开发实例

    至此关于系统资源监控、apache监控调优、Tomcat监控调优、JVM调优、Mysql调优、前端监控调优、接口性能监控调优的知识已分享完,今天学习全链路脚本开发知识。后续文章都会系统分享干货,带大家从0到1学会性能测试。 前面章节介绍了如何封装.h头文件,现在通过一个实例来介绍…

    Spring AI Alibaba + Nacos 动态 MCP Server 代理方案

    作者:刘宏宇,Spring AI Alibaba Contributor 文章概览 Spring AI Alibaba MCP 可基于 Nacos 提供的 MCP server registry 信息,建立一个中间代理层 Java 应用,将 Nacos 中注册的服务信息转换成 MCP 协议的服务器信息&#xff0c…

    MCP:让AI工具协作变得像聊天一样简单 [特殊字符]

    想象一下,你正在处理一个项目,需要从A平台查看团队讨论,从B平台获取客户信息,还要在GitHub上检查代码进度。传统做法是什么?打开三个不同的网页,在各个平台间来回切换,复制粘贴数据,最后还可能因为信息分散而遗漏重要细节。 听起来很熟悉?这正是当前工作流程的痛点所…

    AI炼丹日志-27 - Anubis 通过 PoW工作量证明的反爬虫组件 上手指南 原理解析

    点一下关注吧!!!非常感谢!!持续更新!!! Java篇: MyBatis 更新完毕目前开始更新 Spring,一起深入浅出! 大数据篇 300: Hadoop&…

    阿姆达尔定律的演进:古斯塔夫森定律

    前言 在上一篇文章《使用阿姆达尔定律来提升效率》中提到的阿姆达尔定律前提是假设问题的规模保持不变,并且给定一台速度更快的机器,目标是更快地解决问题。然而,在大多数情况下,这并不完全正确。当有一台更快的机器时&#xff0…

    JavaScript极致性能优化全攻略

    JavaScript性能优化深度指南 1 引言 JavaScript性能优化在现代Web开发中至关重要。随着Web应用日益复杂,性能直接影响用户体验、搜索引擎排名和业务转化率。研究表明,页面加载时间每增加1秒,转化率下降7%,跳出率增加32%。通过优化JavaScript性能,开发者可以: 提升用户满…

    Transformer核心原理

    简介 在人工智能技术飞速发展的今天,Transformer模型凭借其强大的序列处理能力和自注意力机制,成为自然语言处理、计算机视觉、语音识别等领域的核心技术。本文将从基础理论出发,结合企业级开发实践,深入解析Transformer模型的原…

    Grafana-State timeline状态时间线

    显示随时间推移的状态变化 状态区域:即状态时间线上的状态显示的条或带,区域长度表示状态持续时间或频率 数据格式要求(可视化效果最佳): 时间戳实体名称(即:正在监控的目标对应名称&#xf…