RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 568119
Accepted
Arch
Arch
Asked:2020-09-20 02:42:49 +0000 UTC2020-09-20 02:42:49 +0000 UTC 2020-09-20 02:42:49 +0000 UTC

ArrayList 与 LinkedList 的区别?

  • 772

LinkedList与有什么区别ArrayList?而在实际中什么情况下使用起来更方便LinkedList呢?

java
  • 3 3 个回答
  • 10 Views

3 个回答

  • Voted
  1. Best Answer
    GreyGoblin
    2020-09-20T02:59:48Z2020-09-20T02:59:48Z

    ArrayList 是一个基于数组的列表。LinkedList - 基于元素和它们之间的链接的链表。作为 LinkedList,串联连接的火车车厢的表示最适合。

    当索引访问优先时,应使用ArrayList ,因为这些操作是在恒定时间内执行的。添加到列表的末尾,平均而言,也需要常数时间。此外,在 ArrayList 中,在元素之间存储捆绑包不会产生额外成本。插入/删除不在列表末尾的元素的速度不足,因为在此操作期间,添加/删除右侧的所有元素都会移动。

    当插入/删除操作的性能更为重要时, LinkedList很有用,这在 LinkedList 中是在恒定时间内执行的。索引访问操作是通过从开头或结尾(以较近者为准)迭代到所需元素来执行的。在元素之间存储捆绑包的额外成本。

    总之 - 如果你经常插入/删除 - 选择LinkedList,否则ArrayList

    • 51
  2. Artem Konovalov
    2020-10-28T21:33:18Z2020-10-28T21:33:18Z

    ArrayList基于普通数组。这个集合动态增加数组的大小,如果没有足够的空间,调用add(T element)方法时addAll(Collection<T> other)trimToSize()

    LinkedList它是由节点组成的普通链表。每个节点存储指向下一个/上一个节点和值的链接。在列表本身中,有指向最后一个和第一个节点的链接,以及大小。

    为了评估这些数据结构,可以求助于操作的渐近复杂性:

                              |  ArrayList  |  LinkedList 
     add (в начало)           |     O(n)    |   O(1)
     add (в середину)         |     O(n)    |   O(n)
     add (в конец списка)     |     O(n)    |   O(1)   
    

    LinkedList插入按如下方式执行:找到元素,插入的元素必须遵循该元素,更改其中的链接和紧随其后的链接。

    ArrayList如果当前数组中没有空间,则会创建一个新数组。插入元素之前的那些元素保留在原位,或被复制到新元素中。接下来,添加插入的元素。然后复制原始文件中的剩余元素。

    get (первый элемент)        |   O(1)    |   O(1)
    get (из середины)           |   O(1)    |   O(n)
    get (последний элемент)     |   O(1)    |   O(1)
    

    为了LinkedList找到具有所需索引的元素,您需要从第一个元素到最后一个元素(在最坏的情况下)一个一个地遍历链接。ArrayList通过简单地从数组中按索引获取元素来获取元素。

    delete (первый элемент)     |   O(n)    |   O(1)
    delete (из середины)        |   O(n)    |   O(n)
    delete (последний элемент)  |   O(1)    |   O(1)
    

    删除LinkedList类似于插入。在ArrayList中,与添加时大致相同。

    正如我们平均看到的那样,困难是相同的。但我不建议使用LinkedList,除非在列表的开头或结尾删除或插入占主导地位。

    ArrayList就数据布局而言,处理器的可预测性更高。这是一个数组,里面的元素是顺序排列的,占据一块连续的内存区域 。这很好,因为它允许您在没有cache miss'ов. 处理器没有空闲,正在等待来自 RAM 的数据。没有LinkedList这样的事情,因为 元素位于内存的不同部分,处理器无法预测下一个元素的位置。

    显示性能差异的代码:

    @Fork(1)
    @Warmup(iterations = 10)
    @Measurement(iterations = 10)
    @BenchmarkMode(Mode.AverageTime)
    @Threads(1)
    @State(Scope.Benchmark)
    @OutputTimeUnit(TimeUnit.MILLISECONDS)
    public class PerformanceTest {
       private static List<Object> arrayList;
       private static List<Object> linkedList;
    
       private static final int count = 100_000;
    
       public static void main(String[] args) throws Exception {
        Main.main(args);
       }
    
       @Setup
       public static void setup() {
          arrayList = new ArrayList<>(count);
          linkedList = new LinkedList<>();
    
          for (int i = 0; i < count; i++)
             arrayList.add(new Object());
    
          linkedList.addAll(arrayList);
       }
    
       @Benchmark
       public void removeFromLinkedList(Blackhole blackhole) throws Exception {
          Object object = new Object();
          linkedList.remove(count / 2);
          linkedList.add(count / 2, object);
       }
    
       @Benchmark
       public void removeFromArrayList(Blackhole blackhole) throws Exception {
          Object object = new Object();
          arrayList.remove(count / 2);
          arrayList.add(count / 2, object);
       }
    } 
    

    在我的电脑上,情况是这样的:

    Benchmark                             Mode  Cnt  Score    Error  Units
    PerformanceTest.removeFromArrayList   avgt   10  0.011 ±  0.001  ms/op
    PerformanceTest.removeFromLinkedList  avgt   10  0.148 ±  0.001  ms/op
    

    结果表明它LinkedList慢了14倍。

    • 35
  3. Sergey Bespalov
    2020-09-20T11:18:53Z2020-09-20T11:18:53Z

    在选择ArrayList列表作为实现时,应该明白添加元素的操作可能会导致需要增加数组的大小,这将导致将所有元素从一个数组复制到另一个数组的操作。对此要注意初始容量,可以在创建列表时在构造函数中指定public ArrayList(int initialCapacity)。

    如果您无法估计集合中将存储的元素的估计数量并且您不需要通过索引访问元素,那么最好注意LinkedList。

    了解不同类型的集合有何不同非常重要,但在实践中,对于大多数应用程序任务,我们谈论的是数十个和数百个元素,特定集合实现的选择在性能和内存使用情况。更重要的是您将要使用的集合接口的选择:

    • java.util.集合
    • java.util.列表
    • java.util.Set

    接口的选择将直接影响你代码的逻辑。接口还可以更好地用作公共方法的参数和返回类型。

    • 11

相关问题

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    如何停止编写糟糕的代码?

    • 3 个回答
  • Marko Smith

    onCreateView 方法重构

    • 1 个回答
  • Marko Smith

    通用还是非通用

    • 2 个回答
  • Marko Smith

    如何访问 jQuery 中的列

    • 1 个回答
  • Marko Smith

    *.tga 文件的组重命名(3620 个)

    • 1 个回答
  • Marko Smith

    内存分配列表C#

    • 1 个回答
  • Marko Smith

    常规赛适度贪婪

    • 1 个回答
  • Marko Smith

    如何制作自己的自动完成/自动更正?

    • 1 个回答
  • Marko Smith

    选择斐波那契数列

    • 2 个回答
  • Marko Smith

    所有 API 版本中的通用权限代码

    • 2 个回答
  • Martin Hope
    jfs *(星号)和 ** 双星号在 Python 中是什么意思? 2020-11-23 05:07:40 +0000 UTC
  • Martin Hope
    hwak 哪个孩子调用了父母的静态方法?还是不可能完成的任务? 2020-11-18 16:30:55 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    user207618 Codegolf——组合选择算法的实现 2020-10-23 18:46:29 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    Arch ArrayList 与 LinkedList 的区别? 2020-09-20 02:42:49 +0000 UTC
  • Martin Hope
    iluxa1810 哪个更正确使用:if () 或 try-catch? 2020-08-23 18:56:13 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5