lamda获取参数集合去空_集合源码解析之LinkedList
在日常开发中,最常用的List是ArrayList其次便是LinkedList了.上次我们已经研究过了ArrayList,今天来深入学习下LinkedList...
概述
LinkedList顾名思义本质上就是一个链表.它和ArrayList一样实现了List接口.
ArrayList是基于可变数组实现的,因此对于随机访问和修改ArrayList的效率会更高,而LinkedList更擅长于随机插入和删除,毕竟只需要移动"指针"即可.
源码分析
结构图
继承关系
public class LinkedList<E>extends AbstractSequentialList<E>implements List<E>, Deque<E>, Cloneable, java.io.Serializable {
LinkedList
继承了AbstractSequentialList
,同时又实现了List
Deque
Cloneable
Serializable
.
AbstractSequentialList
继承了 AbstractList
是LinkedList的父类,是List的简单实现,提供了对连续访问的支持,对于随机访问数据,应优先使用AbstractList.
为何又实现List
的原因在此就不重复叙述了,如想了解可==点击此处(ArrayList)==.
Deque
继承自Queue
, Queue
是Java中所有队列实现的根接口.
public interface Queue<E> extends Collection<E> { // 把元素插入到队列末端 插入成功返回true, 若无空间可插入,则抛异常 (不推荐使用) boolean add(E e);
// 把元素插入到队列末端 插入成功返回true, 反之 false boolean offer(E e);
// 从队首删除一个元素并返回该元素 若队列为空抛异常 (不推荐使用) E remove();
// 从队首删除一个元素并返回该元素 若队列为空,返回null E poll();
// 获取队首的元素(只是获取并不会删除) 若队列是空的会抛异常 (不推荐使用) E element();
// 获取队首的元素(只是获取并不会删除) 若队列是空,返回null E peek();}
Deque
是一个双端队列同时又提供了对栈的抽象,其支持在两端插入和移除元素.通常Deque的实现对容量是没有固定限制,但此接口即支持限容的双端队列,也支持无限容的双端队列.
public interface Deque<E> extends Queue<E> { void addFirst(E e); // 队首插入元素,插入失败抛异常
void addLast(E e); // 队尾插入元素,插入失败抛异常
boolean offerFirst(E e); // 队首插入元素,插入失败返回false
boolean offerLast(E e); // 队尾插入元素,插入失败返回false
E removeFirst(); // 队首删除元素,删除失败抛异常
E removeLast(); // 队尾删除元素,删除失败抛异常
E pollFirst(); // 队首删除元素, 队列为空抛异常
E getFirst(); // 队首获取(不删除)元素,队列为空抛异常
E peekFirst(); // 队首获取(不删除)元素,队列为空返回null ...}
它的api里提供了两种方式.一种在操作失败时抛出异常,另一种形式返回特殊值(null/false).插入操作的后一种形式是专为使用有容量限制的Deque实现设计的;在大多数实现中,插入操作不能失败.
Deque
包含的栈相关api:
E peek(); // 查看栈顶元素
void push(E e); // 入栈
E pop(); // 弹栈
注意: Java堆栈Stack类已经过时,官方推荐使用Deque代替Stack使用.
/** *
A more complete and consistent set of LIFO stack operations is
* provided by the {@link Deque} interface and its implementations, which
* should be used in preference to this class. For example:
*
{@code * Deque stack = new ArrayDeque();}
*/
链表节点实体
Node是LinkedList维护链表结构的核心私有类,比较简单,直接看代码.
private static class Node<E> { // 元素 E item; // 指向下一个元素的指针 Node next;// 指向上一个元素的指针 Node prev; Node(Node prev, E element, Node next) {this.item = element;this.next = next;this.prev = prev; }}
类中属性
通过first
last
俩节点来维护链表进行各项操作,注意transient
关键字,自定义了序列化方式.
/** 版本号,用于校验正反序列化时的一致性 */private static final long serialVersionUID = 876323262645176354L;
/** 元素数量 **/transient int size = 0;
/** 链表的首节点 */transient Node first;/** 链表的尾节点 */transient Node last;
构造函数
两个构造器.一个无参,一个插入指定集合.
// 无参构造器public LinkedList() {}
// 构造一个包含指定集合元素的LinkedListpublic LinkedList(Collection extends E> c) { this(); // 添加指定集合中的所有元素到LinkedList中 addAll(c);}
核心函数
由于接口实现较多,各个功能的多个实现核心差不多,这里挑选每种功能里核心的来讲解.如想深入每个函数,请自行翻阅源码.
查询节点
查询函数较为简单
/** * 返回列表中指定位置的元素。 */public E get(int index) { // 检查下标是否是链表内的 checkElementIndex(index); // node(index) 会获取到下标为index的节点 return node(index).item;}
Node node(int index) { // 这里通过简单的二分法,判断index于链表中间位置的距离 if (index > 1)) { // 若和中间较近从头部开始遍历 Node x = first;for (int i = 0; i x = x.next;return x; } else {// 否则 从尾节点开始 Node x = last;for (int i = size - 1; i > index; i--) x = x.prev;return x; }}public E getFirst() {final Node f = first;if (f == null)throw new NoSuchElementException();return f.item;}public E getLast() {final Node l = last;if (l == null)throw new NoSuchElementException();return l.item;}
node(index)
函数则是根据下标查找节点.
从这里就可以看出,基于双向链表结构的LinkedList,通过索引index
查询是低效的,index
元素越靠近中间所耗费的实际就越长,而在链表双端进行查询则会非常高效.
删除节点
/** * 删除指定元素 */public boolean remove(Object o) { // LinkedList可以接收null值 if (o == null) { // 元素为null for (Node x = first; x != null; x = x.next) {if (x.item == null) { unlink(x);return true; } } } else {// 元素有值for (Node x = first; x != null; x = x.next) {if (o.equals(x.item)) { unlink(x);return true; } } }return false;}/** * 剔除指定节点,并返回 */E unlink(Node x) {final E element = x.item;final Node next = x.next;final Node prev = x.prev;// 若删除的节点为头结点if (prev == null) { first = next; } else { prev.next = next; x.prev = null; }// 若删除的节点是尾节点if (next == null) { last = prev; } else { next.prev = prev; x.next = null; }// 清空x相关引用,方便gc x.item = null; size--; modCount++;return element;}
通过remove(Object o)
可以看出LinkedList是支持存储null
值的,而unlink(Node x)
就是把x
节点给置空,同时把x
节点的上一个节点和下一个节点关联起来.
修改节点
/** * 用指定的元素替换列表中指定位置的元素。 */public E set(int index, E element) { checkElementIndex(index); // 定位需要修改的节点 Node x = node(index); E oldVal = x.item; x.item = element;return oldVal;}
set
函数看起来比较简单,只要定位到节点,并修改节点内的元素就行了.
添加节点
// 添加元素public boolean add(E e);
// 在指定下标插入元素public void add(int index, E element);
// 在指定下标插入指定集合public boolean addAll(int index, Collection extends E> c);
// 添加指定节点为第一个节点public void addFirst(E e);
// 添加指定节点为尾节点public void addLast(E e);
// 把指定集合里的元素添加到该LinkedList中public boolean addAll(Collection extends E> c);
boolean add(E e)
add(E e)
函数实际上是通过linklast(E e)
来实现.而linklast(E e)
内部也较为简单,即在老的last
节点后再加入一个节点.注释较全,直接看代码
/** * 添加元素 */public boolean add(E e) { linkLast(e); return true;}
/** * 添加指定元素到尾节点 */void linkLast(E e) { final Node l = last;// 创建一个新节点,同时把这个节点的pred设置为原来的尾节点final Node newNode = new Node<>(l, e, null);// 设置当前节点为last节点 last = newNode;// 如果原本无尾节点,说明当前无任何节点if (l == null)// 把这个新的节点同时设定为首节点 first = newNode;else// 设置 老的尾节点的下一个节点为newNode l.next = newNode;// 元素数量+1 , 操作+1 size++; modCount++;}
翻阅源码即可发现类似linkLast(E e)
的函数还有俩,
/** * 添加指定元素为当前的首节点 */ private void linkFirst(E e) { final Node f = first;final Node newNode = new Node<>(null, e, f); first = newNode;if (f == null) last = newNode;else f.prev = newNode; size++; modCount++; }/** * 在指定节点的位置插入节点 */void linkBefore(E e, Node succ) {final Node pred = succ.prev;final Node newNode = new Node<>(pred, e, succ); succ.prev = newNode;if (pred == null) first = newNode;else pred.next = newNode; size++; modCount++; }
其中linkFirst(E e)
和linkLast(E e)
没啥区别,无非是把last
换成了first
.
而linkBefore
则是把succ
的上一个节点的next
指向新节点,同时把新节点的next
指向succ
,再把succ
的prev
指向新节点,如下图:
void add(int index, E element)
该函数用到了上述讲过的 linkLast(element)
和linkBefore(element)
函数.
/** * 在指定的下标插入节点 */public void add(int index, E element) { checkPositionIndex(index);
if (index == size) linkLast(element); else linkBefore(element, node(index));}
private void checkPositionIndex(int index) { if (!isPositionIndex(index)) throw new IndexOutOfBoundsException(outOfBoundsMsg(index));}
private boolean isPositionIndex(int index) { return index >= 0 && index <= size;}
先调用checkPositionIndex
检测插入下标是否越界,再根据index
是否是size
来判断调用的插入函数.
boolean addAll(int index, Collection c);
/** * 从指定下标开始插入指定的集合 */public boolean addAll(int index, Collection extends E> c) { // 越界检查 checkPositionIndex(index);
// 转换数组 Object[] a = c.toArray(); int numNew = a.length; if (numNew == 0) return false;
// 获取 index索引位置上的节点,和其上一个节点 Node pred, succ;if (index == size) { succ = null; pred = last; } else { succ = node(index); pred = succ.prev; }// 从index位置开始插入for (Object o : a) { E e = (E) o; Node newNode = new Node<>(pred, e, null);if (pred == null) first = newNode;else pred.next = newNode; pred = newNode; }// 最后把原位置上节点的prev设置为集合c的最后一个节点if (succ == null) { last = pred; } else { pred.next = succ; succ.prev = pred; }// 修改计数 size += numNew; modCount++;return true;}
addAll(int index, Collection extends E> c)
的代码首先是把集合c
转换成Object[]
,然后获取index
索引上的节点,再以其上一个节点为起点开始插入.
总结
只要理解了双向链表的存储结,再看LinkedList的源码就会简单很多了.LinkedList其余函数在此就不叙述了,实现上相似.源码分析的差不多了,这里来做下总结.
LinkedList是一个有序的可重复允许null值的集合,看其内部实现不存在容量不足的问题.底层使用了双向链表结构维护了first和last指针.实现了栈和队列相关接口,所以可做栈,队列,双端队列来使用.
lamda获取参数集合去空_集合源码解析之LinkedList相关推荐
- Java集合框架之三:HashMap源码解析
Java集合框架之三:HashMap源码解析 版权声明:本文为博主原创文章,转载请注明出处,欢迎交流学习! HashMap在我们的工作中应用的非常广泛,在工作面试中也经常会被问到,对于这样一个重要的集 ...
- Spring5源码 - 13 Spring事件监听机制_@EventListener源码解析
文章目录 Pre 概览 开天辟地的时候初始化的处理器 @EventListener EventListenerMethodProcessor afterSingletonsInstantiated 小 ...
- lamda获取参数集合去空_JAVA集合框架知识
1. Vector用法和ArrayList区别 (1) Vector的特有方法有哪些? void addElement(E obj) 将指定的组件添加到此向量的末尾,将其大小增加1. (2) Vect ...
- Java集合框架之 Java HashMap 源码解析
继上一篇文章Java集合框架综述后,今天正式开始分析具体集合类的代码,首先以既熟悉又陌生的HashMap开始. 签名(signature) public class HashMap<K,V> ...
- python 解包_【源码解析】python解包操作一文完全理解
解包是如何操作? >>> a, b = [1, 2] # 以下为此解包操作的字节码 0 LOAD_CONST 1 (1) 2 LOAD_CONST 2 (2) 4 BUILD_LIS ...
- lamda获取参数集合去空_(转)Java8使用lambda表达式进行集合的遍历
本文转自 我们经常会用到各种集合,数字的,字符串的还有对象的.它们无处不在,哪怕操作集合的代码要能稍微优化一点,都能让代码清晰很多.在这章中,我们探索下如何使用lambda表达式来操作集合.我们用它来 ...
- 获取腾讯视频真实地址php源码,解析腾讯视频真实地址 - osc_hajrc28s的个人空间 - OSCHINA - 中文开源技术交流社区...
解析腾讯视频真实地址 标签(空格分隔): php 分析 1 我们从腾讯视频上找到的网页链接格式是这样的 https://v.qq.com/x/page/b0136et5ztz.html vid = b ...
- [源码解析] 机器学习参数服务器 Paracel (1)-----总体架构
[源码解析] 机器学习参数服务器 Paracel (1)-----总体架构 文章目录 [源码解析] 机器学习参数服务器 Paracel (1)-----总体架构 0x00 摘要 0x01使用 1.1 ...
- Java 集合系列03之 ArrayList详细介绍(源码解析)和使用示例
转载自 Java 集合系列03之 ArrayList详细介绍(源码解析)和使用示例 第1部分 ArrayList介绍 ArrayList简介 ArrayList 是一个数组队列,相当于 动态数组.与 ...
最新文章
- Tomcat 1099端口占用重启无效,查不到进程,改换端口无效解决方案
- BZOJ 1091([SCOI2003]分割多边形-分割直线)
- VS 2008 mfc 智能应用程序 调用 C# webservice.(二)(gsoap+wince)
- C语言 静态库制作(亲测可行)
- SpringCloud Greenwich(四)注册中心之eureka、Zuul和 gateway网关配置
- 发布任务PHP源码,Thinkphp5新威客任务平台源码
- 改名之后的 Java EE,现在有什么新进展?
- github提交代码403
- iOS底层探索之类的加载(三): attachCategories分析
- 一周学会php 视频,一周学会PHP视频 普通下载
- java实现身份证识别
- 室友转行软件测试,月薪1.4万,天天摸鱼没事干
- 旧计算机硬盘驱动器可以进口吗,解决方案:可以在新计算机上安装旧的机械硬盘驱动器吗?台式计算机添加新旧机械硬盘+分区教程...
- jsp调整字体大小font_html font标签如何设置字体大小?
- FFmpeg无损转换ts为mp4
- push_back、emplace_back、std::move
- 程控交换机与集团电话的区别是什么
- SEASKY开源机械键盘
- 终于来了!“微信小号”全面开放注册!
- 【隧道篇 / PPTPL2TP】(5.2) ❀ 01. PPTP L2TP 连接 ❀ FortiGate 防火墙
热门文章
- vivox50pro鸿蒙系统,小米10至尊纪念版和vivox50pro+哪一款手机好
- hid在linux上的轮训时间,linux 自定义hid速度优化
- cation,validation,qualification有何区别
- 【译】你不知道的 Chrome 调试工具技巧 第二十天:Workspace的黑魔法
- angularJS1.6.3个人理解(后续更新4.4.7)
- 2.4. myisamchk — MyISAM Table-Maintenance Utility
- windows下端口占用解决方法-查看和杀死占用端口进程
- HDU 3695 Computer Virus on Planet Pandora (AC自己主动机)
- Python 中xrange和range区别
- as3 htmlText 的bug