我们都知道LinkedList和ArrayList相比:

  1、LinkedList插入删除相对较快,而查询较慢;

  2、ArrayList插入删除相对较慢,而查询很快(详细可查看从源码的角度分析List与Set的区别);

由类的关系可知,两者的顶层是一致的,但LinkedList额外的继承了AbstractSequentialList类并实现了Deque接口,也就是说LinkedList拥有双端 队列的功能,LinkedList的实现原理与ArrayList不同,LinkedList是链表的方式实现的

一、LinkedList插入删除相对较快,而查询较慢 LinkedList的add(E e)方法实际调用的是内部的linkLast(E e)方法,也就是添加到队列尾部

//每个元素在LinkedList中的存储结构,item为自身,prev为上一个节点,next为下一个节点
private static class Node<E> {
E item;
Node<E> next;
Node<E> prev; Node(Node<E> prev, E element, Node<E> next) {
this.item = element;
this.next = next;
this.prev = prev;
}
}
//添加元素
public boolean add(E e) {
linkLast(e);
return true;
}
//添加元素到链表尾部
void linkLast(E e) {
final Node<E> l = last;
//创建一个链表节点,上个元素为原有链表的最后一个元素,下一个元素为空
final Node<E> newNode = new Node<>(l, e, null);
//设置新的元素为链表最后一个元素
last = newNode;
//如果原来最后一个元素为空,则当前元素时第一个元素,给第一个元素的变量赋值
//如果原来的最后一个元素存在,则修改该元素的下一个元素为新的元素
if (l == null)
first = newNode;
else
l.next = newNode;
size++;
modCount++;
}

LinkedList的add(int index,E e)方法是在某个索引前插入元素E,先根据索引index找到相应元素,在该元素前添加新元素,添加方法与linkLast类似

//在指定位置插入元素
public void add(int index, E element) {
//校验索引是否有效,主要判断索引是否在size的范围内
checkPositionIndex(index);
//如果索引正好等于size,则添加到链表尾部,否则在指定位置插入新元素,修改原来该位置的元素的上级节点为新元素,新元素的上级元素为原有元素的上级元素
if (index == size)
linkLast(element);
else
//node(index)中主要是结合index和size循环找出原有index位置的元素
linkBefore(element, node(index));
}
//在某个节点前插入元素
void linkBefore(E e, Node<E> succ) {
// assert succ != null;
final Node<E> pred = succ.prev;
final Node<E> newNode = new Node<>(pred, e, succ);
succ.prev = newNode;
if (pred == null)
first = newNode;
else
pred.next = newNode;
size++;
modCount++;
}

addFirst、addLast等元素与以上实现类似,作用是添加元素到链表头、链表尾 移除元素remove(Object o)/remove(int index0)与插入类似

//移除元素
public boolean remove(Object o) {
//从第一个元素开始往下循环,找到第一个匹配到的元素移除
if (o == null) {
for (Node<E> x = first; x != null; x = x.next) {
if (x.item == null) {
//把当前元素的上个元素的next设置为当前元素的下一个元素,把当前元素的下一个节点的prev设置为当前元素的上一个元素,并把当前元素的prev、next、item都置为null以便于GC
unlink(x);
return true;
}
}
} else {
for (Node<E> x = first; x != null; x = x.next) {
if (o.equals(x.item)) {
unlink(x);
return true;
}
}
}
return false;
}
//根据索引移除元素,先根据索引找出当前元素再进行移除
public E remove(int index) {
checkElementIndex(index);
return unlink(node(index));
}

根据以上可知,LinkedList的添加和移除等操作只对相应位置的上下元素进行改动即可,相对于ArrayList需要进行数组的复制移位性能要好些。而在查询来说,LinkedList需要根据索引以及内部存在的size、first、last进行循环匹配,相对于ArrayList只需要获取数组的相应下标所对应的值就会相应慢些。

//根据索引获取对象
public E get(int index) {
checkElementIndex(index);
return node(index).item;
}
//根据索引循环查询对应的元素
Node<E> node(int index) {
if (index < (size >> 1)) {
Node<E> x = first;
for (int i = 0; i < index; i++)
x = x.next;
return x;
} else {
Node<E> x = last;
for (int i = size - 1; i > index; i--)
x = x.prev;
return x;
}
}

LinkedList与ArrayList的区别的更多相关文章

  1. LinkedList和ArrayList的区别/何时使用LinkedList和ArrayList

    1.ArrayList是实现了基于动态数组的数据结构,LinkedList基于链表的数据结构. 2.对于随机访问get和set,ArrayList觉得优于LinkedList,因为LinkedList ...

  2. LinkedList和ArrayList的区别

    LinkedeList和ArrayList都实现了List接口,但是它们的工作原理却不一样.它们之间最主要的区别在于ArrayList是可改变大小的数组,而LinkedList是双向链接串列(doub ...

  3. Android LinkedList和ArrayList的区别

    LinkedeList和ArrayList都实现了List接口,但是它们的工作原理却不一样.它们之间最主要的区别在于ArrayList是可改变大小的数组,而LinkedList是双向链接串列(doub ...

  4. Java 中 LinkedList 和 ArrayList 的区别

    引自:https://www.cnblogs.com/huzi007/p/5550440.html ArrayList和LinkedList的大致区别如下:1.ArrayList是实现了基于动态数组的 ...

  5. LinkedList与ArrayList的区别(内部实现)

    ArrayList的内部实现是基于内部数组Object[],所以从概念上讲,它更像数组: LinkedList的内部实现是基于一组连接的记录,所以,它更像一个链表结构,所以,它们在性能上有很大的差别. ...

  6. Java ArrayList和Vector、LinkedList与ArrayList、数组(Array)和列表集合(ArrayList)的区别

    ArrayList和Vector的区别ArrayList与Vector主要从二方面来说.  一.同步性:   Vector是线程安全的,也就是说是同步的,而ArrayList是线程序不安全的,不是同步 ...

  7. 数组Array和列表集合ArrayList、LinkedList和Vector的区别

    一.ArrayList和Vector的区别 ArrayList与Vector主要从以下方面来说. 1.同步性: Vector是线程安全的,也就是说是同步的,而ArrayList是线程序不安全的,不是同 ...

  8. 集合中list、ArrayList、LinkedList、Vector的区别、Collection接口的共性方法以及数据结构的总结

    List (链表|线性表) 特点: 接口,可存放重复元素,元素存取是有序的,允许在指定位置插入元素,并通过索引来访问元素 1.创建一个用指定可视行数初始化的新滚动列表.默认情况下,不允许进行多项选择. ...

  9. java中List接口的实现类 ArrayList,LinkedList,Vector 的区别 list实现类源码分析

    java面试中经常被问到list常用的类以及内部实现机制,平时开发也经常用到list集合类,因此做一个源码级别的分析和比较之间的差异. 首先看一下List接口的的继承关系: list接口继承Colle ...

随机推荐

  1. PHP多个进程同时写入同一个文件

    flock (PHP 3 >= 3.0.7, PHP 4, PHP 5) flock -- 轻便的咨询文件锁定 说明 bool flock ( int handle, int operation ...

  2. Asp.Net初学小结

    第一章   1.搭建Asp.net开发环境   1).net FrameWork(VS) 2)IIS(xp:5.1,2003:6.0,vista:70,win7:7.5) C:\Windows\Mic ...

  3. Yarn的运行原理(执行流程)

    服务功能 ResouceManager:     1.处理客户端的请求     2.启动和监控ApplicationMaster     3.监控nodemanager     4.资源的分配和调度 ...

  4. Java框架之Spring(二)

    前一篇博客讲述了Spring的一些基础概念,下面我们来创建第一个Spring程序吧. 步骤如下: 1) 导包 2) 配置文件 附没有提示的情况 MyEclipse ->File and Edit ...

  5. 【Java基础】2、Java中普通代码块,构造代码块,静态代码块区别及代码示例

    Java中普通代码块,构造代码块,静态代码块区别及代码示例.Java中普通代码块,构造代码块,静态代码块区别及代码示例 执行顺序:静态代码块>静态方法(main方法)>构造代码块>构 ...

  6. 10个JavaScript难点

    译者按: 能够读懂这篇博客的JavaScript开发者,运气不会太差... 原文: 10 JavaScript concepts every Node.js programmer must maste ...

  7. jQ效果:jQuery时间轴插件jQuery Timelinr

    前言 这是一款可用于展示历史和计划的时间轴插件,尤其比较适合一些网站展示发展历程.大事件等场景.该插件基于jQuery,可以滑动切换.水平和垂直滚动.支持键盘方向键.经过扩展后可以支持鼠标滚轮事件. ...

  8. 【读书笔记】iOS-xib,点击事件的连接(三)

    紧接着上一节来写 一,选中On按钮,同时按住Control键,连接到FirstViewController.h文件中. 会弹出如下对话框. 二,将Connection处选择为Action,同时将Nam ...

  9. Intent调用常见系统组件

    // 调用浏览器 Uri webViewUri = Uri.parse("http://blog.csdn.net/zuolongsnail"); Intent intent = ...

  10. Java并发编程(十)阻塞队列

    使用非阻塞队列的时候有一个很大问题就是:它不会对当前线程产生阻塞,那么在面对类似消费者-生产者的模型时,就必须额外地实现同步策略以及线程间唤醒策略,这个实现起来就非常麻烦.但是有了阻塞队列就不一样了, ...