【Java面试】ArrayList、LinkedList 查找数据哪个快

news2025/1/11 15:45:54

ArrayList、LinkedList查找数据哪个快
这里有几种不同情况
1、是不是有序的?
2、说的查找是什么意思?是调用get(1),还是调用的contains(o)方法?

根据上面的问题,我们可以分开讨论:

1、数据是有序的

指定下标查询:
ArrayList 因为是基于数组实现,所以可以随机访问,时间复杂度是O(1);
LinkedList 因为是基于链表实现,所以只能从头到尾遍历,时间复杂度是O(n);

查找元素:
ArrayList 在有序的集合上,可以进行二分查找,时间复杂度是O(logn);
LinkedList 只能从头遍历,时间复杂度是O(n);

所以:ArrayList 比LinkedList 快

2、数据是无序的

指定下标查询:同上

查找元素:
ArrayList 因为无序,只能从头遍历查找,时间复杂度最坏是O(n);
LinkedList 只能从头遍历,时间复杂度最坏是O(n);

从理论上看是一样快,但是你可能忽略了一点,这也是我为什么写这篇博客的原因,我想这种级别的题目,面试官怎么还好意思问,可能并没有我想的这么简单。

猜测:数组中的元素在内存中是连续存储的,而链表中的元素是分散存储在内存中的,必须通过遍历链表来查找特定元素。

面试官是不是想让我回答这个?
于是我去测试了一下:


import java.util.ArrayList;
import java.util.LinkedList;

/**
 * @author yangbin
 * @since 2023-04-22
 */
public class SearchPerformanceTest {
    public static void main(String[] args) {
        ArrayList<Integer> arrayList = new ArrayList<>();
        LinkedList<Integer> linkedList = new LinkedList<>();

        // 将100000个随机整数添加到列表中
        for (int i = 0; i < 100000; i++) {
            int randomNum = (int) (Math.random() * 100000);
            arrayList.add(randomNum);
            linkedList.add(randomNum);
        }

        // 在ArrayList中查找一个存在的元素,并记录时间
        long start = System.currentTimeMillis();
        for (int i = 0; i < 100000; i++) {
            arrayList.contains(50000);
        }
        long end = System.currentTimeMillis();
        System.out.println("ArrayList 查找耗时:" + (end - start) + " 毫秒");

        // 在LinkedList中查找一个存在的元素,并记录时间
        start = System.currentTimeMillis();
        for (int i = 0; i < 100000; i++) {
            linkedList.contains(50000);
        }
        end = System.currentTimeMillis();
        System.out.println("LinkedList 查找耗时:" + (end - start) + " 毫秒");
    }
}

结果如下:
在这里插入图片描述

总结:相同时间复杂度的情况下,ArrayList仍然比LinkedList快1倍左右,主要有内存的存储结构的优势。

那为什么连续内存遍历就更快呢?为什么?为什么?

这是因为计算机在读取内存时,通常是按照连续的内存地址一块一块地读取数据的,这个过程被称为"连续访问"。这样可以利用计算机缓存(cache)的特性,提高内存读取速度。而对于分散存储在内存中的链表节点,由于它们的内存地址是不连续的,因此在查找特定元素时需要一个一个地遍历链表节点,这被称为"随机访问"。相比较而言,随机访问的速度较慢,因为每次查找都需要从内存中读取不同的数据块,而这些数据块可能不在缓存中,需要从主存中读取,从而导致了效率的下降。

而对于ArrayList来说,因为其中的元素是连续存储的,因此可以通过索引直接访问特定的元素,这被称为"顺序访问"。相比较而言,顺序访问比随机访问更快,因为在顺序访问的过程中,计算机可以预先将下一个数据块加载到缓存中,提高读取速度。

因此,当我们需要频繁地访问某些元素时,内存连续的ArrayList会更快,而当我们需要频繁地进行插入、删除等操作时,分散存储在内存中的LinkedList会更适合,因为它们在插入、删除操作时的性能较好。

连续访问为什么可以利用缓存(cache)的特性?为什么?为什么?

计算机中的缓存(cache)是一个小而快速的内存区域,用于存储最近访问的数据。当计算机需要读取内存中的数据时,它首先会检查缓存中是否已经存在这个数据,如果存在,则可以直接从缓存中读取,否则需要从内存中读取数据。由于缓存是比内存快的,因此从缓存中读取数据的速度比从内存中读取数据的速度快得多。

当计算机读取一块内存中的数据时,通常会一次读取多个连续的数据块并将它们存储在缓存中。这样,如果后续需要访问这些数据块,计算机可以直接从缓存中读取,而不需要再次从内存中读取。这就是连续访问的优势,因为连续访问的数据通常是存储在相邻的内存地址中的,计算机可以一次性读取多个数据块并将它们存储在缓存中,从而利用缓存的特性提高读取速度。

相比之下,随机访问的数据通常是分散存储在内存中的,因此计算机需要多次从内存中读取数据块,这会导致缓存中的数据被频繁替换,从而降低缓存的效率。这就是为什么连续访问比随机访问更能够利用缓存的特性的原因。

看到这里,你应该都明白了吧

再扩展一点

为什么计算机读取内存时是按数据块读取?不能只读我这个地址的数据吗? 这里有一个内存页概念:

在计算机中,内存通常被划分为许多固定大小的块,这些块被称为内存页(memory page),一般大小为4KB或8KB。当计算机读取内存中的数据时,它并不是按照单个地址读取数据,而是按照内存页的大小一块一块地读取数据。

这种按照内存页大小读取数据的方式是由计算机的硬件实现的,它有以下几个原因

提高读取速度: 由于内存页是固定大小的,因此计算机可以预先将多个内存页加载到缓存中,这样可以提高内存读取速度,因为如果下一个需要读取的数据在缓存中已经存在,计算机就可以直接从缓存中读取,而不需要再次访问内存。

简化地址转换: 当计算机执行指令时,需要将逻辑地址(logical address)转换成物理地址(physical address),这个过程是由内存管理单元(memory management unit)完成的。按照内存页的方式读取数据可以简化地址转换的过程,因为内存页的大小是固定的,计算机可以直接将逻辑地址映射到物理地址。

提高内存利用率: 按照内存页的方式读取数据可以避免浪费内存空间。如果计算机只按照单个地址读取数据,那么一些内存页可能只有部分被使用,而其他部分则没有被使用,这会导致内存利用率的降低。

总之,按照内存页的方式读取数据可以提高读取速度、简化地址转换和提高内存利用率,这是计算机系统设计中的一项重要技术。

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

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

相关文章

Apifox自动生成接口文档

1、安装 1.1 Apifox安装 官方文档&#xff1a;Apifox - API 文档、调试、Mock、测试一体化协作平台 - 接口文档工具&#xff0c;接口自动化测试工具&#xff0c;接口Mock工具&#xff0c;API文档工具&#xff0c;API Mock工具&#xff0c;API自动化测试工具 1.2 IDEA 插件安装…

Vue 复学 之 状态管理 Vuex

Vuex是vue中的一种状态管理模式&#xff0c;就是一个 状态仓库&#xff0c;仓库做什么&#xff1f;存储状态、管理状态&#xff08;数据&#xff09;的变化、提供状态获取窗口。 本文中一些测试用例基于vue/composition-api1.7.1 &#xff0c; vuex3.6.2&#xff0c; vue2.6.1…

【unity实战】随机地下城生成1——随机生成地下城初稿(含源码)

先看看实现的最终效果 #用到的素材 https://download.csdn.net/download/qq_36303853/87712757 导入素材 导入房间图片素材,配置图片信息信息 点击sprite Editor,开始切割图片 随机创建基本房间 已一个白底图片模拟房间预设体 思路:建立一个空的 GameObject 用来做…

C++中的STL容器

文章目录 一、序列式容器1.vector2.array3.deque4.list5.forward_list 二、关联式容器1.set、multiset、unordered_set和unordered_multiset2.map、multimap、unordered_map和unordered_multimap STL中的容器将一些应用最为广泛的数据结构实现了出来&#xff0c;它主要分为序列…

使用python下载wallpaper Engine订阅的壁纸/视频

一、为什么想下载wallpaper Engine的壁纸 在游戏平台steam上&#xff0c;有一个壁纸软件wallpaper Engine&#xff0c;人称小红车&#xff0c;里面有各种好看的动态壁纸和视频&#xff0c;可以给我们的电脑设置动态桌面&#xff0c;非常好用。   用过几次后&#xff0c;我有了…

Doris(14):索引

1 概念 索引用于帮助快速过滤或查找数据。 目前 Doris 主要支持两类索引&#xff1a; 内建的智能索引&#xff0c;包括前缀索引和ZoneMap索引用户创建的二级索引&#xff0c;包括Bloom Filter索引和Bitmap倒排索引。 前缀索引&#xff1a;即在排序的基础上&#xff0c;实现的…

kv server(配置以及性能测试)

首先在 Cargo.toml 里添加 serde 和 toml。我们计划使用 toml 做配置文件&#xff0c;serde 用来处理配置的序列化和反序列化&#xff1a; [dependencies] ... serde { version "1", features ["derive"] } # 序列化/反序列化 ... toml "0.5"…

antd表格a-table滚动失效。x轴滚动失效

目录 antd表格a-table滚动失效。x轴滚动失效 页面html代码如下。实现左右布局&#xff0c;左边侧边栏固定宽度&#xff0c;右边沾满剩余宽度 解决方案&#xff1a;在计算右侧宽度时&#xff0c;左边侧边栏固定宽度&#xff0c;右边沾满剩余宽度 情况1&#xff1a;左侧侧边栏…

第八章 查询和检索:Query DSL

版权声明 本文为Elastic开源社区版权所有,保证独立性和原创性,未获得授权和允许,任何组织和个人不得以任何方式传播或复制或分享。否则必将追究法律责任。 知识内容输出不易,请尊重他人劳动成果。严禁随意传播、复制和盗用他人成果或文章内容用以商业或盈利目的! 1、查询…

5.4 龙贝格算法

为什么有龙贝格算法&#xff1a; 龙贝格算法是一种数值积分方法&#xff0c;用于计算定积分的数值近似值。它是基于复合梯形法和复合辛普森法的推广和拓展&#xff0c;可以达到更高的精度。相较于复合梯形法和复合辛普森法&#xff0c;龙贝格算法的收敛速度更快&#xff0c;且…

nerfstudio介绍及在windows上的配置、使用

nerfstudio提供了一个简单的API&#xff0c;可以简化创建、训练和可视化NeRF的端到端过程。该库通过模块化每个组件来支持可解释的NeRF实现。nerfstudio源码地址: https://github.com/nerfstudio-project/nerfstudio , 通过模块化集成了多个NeRF扩展的实现&#xff0c;持续更新…

JUC-多线程(12. AQS)学习笔记

文章目录 1. 可重入锁1.1. 概述1.2. 可重入锁类型1.3. Synchronized 可重入实现机理 2. LockSupport2.1. LockSupport 是什么2.2. 3种线程等待唤醒的方法2.2.1 Object 的等待与唤醒2.2.2. Condition接口中的等待与唤醒2.2.3. 传统的 synchronized 和 Lock 实现等待唤醒通知的约…

C/C++开发,opencv读写图像与视频

目录 一、opencv的图像缓存表达&#xff08;cv::mat&#xff09; 二、图片读写 2.1 图片读写API 2.2 图片读写案例 2.3 案例编译与测试 三、opencv的视频读写&#xff1a; 3.1 视频读写接口 3.2 视频读写案例 3.3 编译与测试 一、opencv的图像缓存表达&#xff08;cv::mat&am…

2023网络搭建项目改革

好久没更新了&#xff0c;哈哈哈&#xff0c;也废话不多说&#xff0c;直接进入正题。 3月的时候就有人吓我说什么网搭取消了&#xff0c;当时我还觉得高兴&#xff0c;主要是不喜欢这个行业&#xff0c;要是没了我就可以跑路了&#xff0c;哈哈&#xff0c;然后我就觉得很奇怪…

【嵌入式系统与入门】Day02 Arduino 按键、蜂鸣器与湿温度传感器

文章目录 1. 按键控制1.1 认识按键1.2 工作原理1.3 Arduino代码展示1.4 原理图1.5 实现去抖【消抖动延时】 2. 蜂鸣器控制2.1 认识蜂鸣器2.2 分类2.3 工作原理2.4 连线2.5 Arduino代码展示 3. PWM模拟量输出3.2 Arduino代码展示 4. 湿温度测量4.1 认识器件4.2 传感器接口4.3 Ar…

CMU-自主探索导航系统(TARE FAR Planner)学习-All in one

参考引用 Autonomous Exploration Development EnvironmentTARE机器人自主导航系统社区-CSDN社区云TARE机器人自主导航系统公开课1TARE机器人自主导航系统公开课2CMU团队开发的全套开源自主导航算法FAR Planner —— IROS2022 最佳学生论文&#xff1c;论文阅读&#xff1e;TAR…

【源码分析】XXL-JOB的执行器的注册流程

目的&#xff1a;分析xxl-job执行器的注册过程 流程&#xff1a; 获取执行器中所有被注解(xxlJjob)修饰的handler执行器注册过程执行器中任务执行过程 版本&#xff1a;xxl-job 2.3.1 建议&#xff1a;下载xxl-job源码&#xff0c;按流程图debug调试&#xff0c;看堆栈信息…

【ONE·C++ || stack queue (一)】

总言 主要介绍栈和队列的基本函数使用&#xff1a;栈和队列、优先级队列、适配器、反向迭代器。 文章目录 总言1、栈和队列接口基本介绍1.1、基本介绍1.2、相关例题1.2.1、最小栈1.2.2、栈的压入、弹出序列1.2.3、逆波兰表达式求值 2、适配器介绍2.1、引入&#xff1a;如何实现…

儿童用灯哪个品牌好?分享五款儿童护眼台灯品牌

家中有小朋友上了幼儿园就已经戴上了眼镜&#xff0c;太让人心疼了 近视已经成为世界难题&#xff0c;而我国儿童近视形式尤为严峻 据官方数据显示&#xff0c;我国儿童青少年总体近视率竟高达52.7% 如何保护孩子眼睛&#xff0c;儿童用灯哪个品牌好&#xff1f; 那今天&am…

Open vSwitch 入门实践(8) VXLAN实验

目录 什么是VXLAN&#xff1f; VXLAN解决了什么问题&#xff1f; VXLAN网络如何工作&#xff1f; 简单VXLAN实验 主机A 主机B 测试 什么是VXLAN&#xff1f; VXLAN&#xff08;Virtual eXtensible Local Area Network&#xff0c;虚拟扩展局域网&#xff09;&#xff0…