LinkList源码浅析
时间:2019-09-28
本文章向大家介绍LinkList源码浅析,主要包括LinkList源码浅析使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
public class LinkedList<E>
extends AbstractSequentialList<E>
implements List<E>, Deque<E>, Cloneable, java.io.Serializable
{
//实际的存储对象的数量,transient的作用是使得改成员变量不会被序列化
transient int size = 0;
//第一个结点
transient Node<E> first;
//最后一个结点
transient Node<E> last;
//内部结点类,它的实例对象就是双向链表的一个结点
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 LinkedList() {
}
//利用一个容器对象进行初始化
public LinkedList(Collection<? extends E> c) {
this();
addAll(c);
}
//上面的有参构造调用该方法,当然也可以直接使用它复制一个容器进入linklist
public boolean addAll(Collection<? extends E> c) {
//从size也就是当前最后一个结点的的位置+1的位置添加,因为下标是从0开始
return addAll(size, c);
}
//从一个固定位置复制一个容器进入linklist
public boolean addAll(int index, Collection<? extends E> c) {
//判断代码为 index >= 0 && index <= size; 判断这个给的这个位置是否合理
//不是这个范围会抛出数组越界异常
checkPositionIndex(index);
//将容器转换为数组
Object[] a = c.toArray();
int numNew = a.length;
//如果要添加的容器是空的,直接返回一个false
if (numNew == 0)
return false;
//succ是index所在位置的那个结点的,pred是succ上一个结点,
Node<E> pred, succ;
if (index == size) {
//如果index和size相等,说明这个就是要在linklist的末尾后面添加
//所以succ应该赋值为空,pred就是末尾的那个结点
succ = null;
pred = last;
} else {
//否则说明index那个位置不是空的,就把那个位置的结点的引用给succ
//node(index)就是遍历这个双向链表,返回index位置结点的引用
succ = node(index);
pred = succ.prev;
}
//前面的代码确定了要添加的位置,这里开始遍历容器转换的数组,进行添加
for (Object o : a) {
@SuppressWarnings("unchecked") E e = (E) o;
//生成一个结点,注意pred作为该结点的前一个结点,之后不用进行e.prev = prev,因为构造方法已经完成了这一步
Node<E> newNode = new Node<>(pred, e, null);
if (pred == null)
//如果pred是空的,说明整个linklist是空的
first = newNode;
else
//linklist是双向链表,前一个结点需要说明它的下一个结点是谁
pred.next = newNode;
//这一步让pred指向新的结点,完成了一个结点的添加
//注意:我们是让pred指向要添加的位置的的前一个结点,现在要添加的位置应该是新结点所在位置的下一个位置了
pred = newNode;
}
//上面结束遍历后,容器是复制完成了,但是没添加之前,index位置之后的链表的其余部分需要添加回来
//succ现在应该让它的前一个结点是新结点,新结点的下一个结点应该是succ
if (succ == null) {
//如果succ是空的,说明我们是在链表的末尾的下一个位置添加的,last需要重新指向正确的结点,也就是当前的最后一个结点
//这个pred经过上面的遍历,指向的是最后添加的那个结点,就是最后一个结点
last = pred;
} else {
//如果succ不是空的,当然原先是末尾的结点现在仍然是末尾,所以就不用last = pred;
//这个pred经过上面的遍历,指向的是最后添加的那个结点,重要的事情说两遍
//这里是succ和pred建立联系
pred.next = succ;
succ.prev = pred;
}
//更改现在linklist中存在的结点对象的个数
size += numNew;
//记录修改次数,使用迭代器时使用linklist的增删改会出现并发修改异常
//关于modCount在另一篇浅析arraylist做了些说明,就不赘述了
modCount++;
return true;
}
//在linklist内部总调用的方法,因为总是需要得到某个位置的结点引用
Node<E> node(int index) {
//这一句不是我注释的,其代码为index >= 0 && index < size,判断下范围,和之前checkPositionIndex作用差不多
// assert isElementIndex(index);
//size >> 1时将size除以2,但是这个是直接把指令给cpu的,是最快的运算
//除2是为了更快一点,这样最多只用遍历一半长度的链表就能得到index位置的结点
//遍历方式就是朝着一个方向一直取next或者prev,直到到达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;
}
}
//添加一个元素的方法,知道了addAll(int index, Collection<? extends E> c),添加一个元素看的就容易了
public boolean add(E e) {
linkLast(e);
return true;
}
//真正的添加一个元素的方法
void linkLast(E e) {
final Node<E> l = last;
//让新结点的前一个结点是linklist的最后一个结点
final Node<E> newNode = new Node<>(l, e, null);
//last指向作为最后一个结点的新结点newNode
last = newNode;
if (l == null)
//l为旧的last,如果l为空说明链表是空的,说明first需要指向作为第一个结点的新结点newNode
first = newNode;
else
//l不为空,那么l现在是倒数第二个结点,它需要知道它的下一个结点是谁
l.next = newNode;
//更改当前linklist存储的对象的个数
size++;
//更改修改次数
modCount++;
}
//删除一个位置的元素的方法
public E remove(int index) {
//检测index是否合理
checkElementIndex(index);
return unlink(node(index));
}
//删除一个结点的方法
E unlink(Node<E> x) {
//这句是官方注释
// assert x != null;
//该结点存储的元素对象
final E element = x.item;
final Node<E> next = x.next;
final Node<E> prev = x.prev;
//这下面的两个if else就是想让该结点的前一个和后一个结点相互建立联系,消除该结点它们的联系
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.item这个引用,让垃圾回收器回收这个被删除的结点的那块内存
x.item = null;
size--;
modCount++;
return element;
}
//根据对象删除一个linklist中元素对象的方法
public boolean remove(Object o) {
//总之从头结点开始遍历,找到那个存储元素对象等于要删除的对象的结点
//调用上面的 E unlink(Node<E> x)方法,删除该结点
if (o == null) {
for (Node<E> x = first; x != null; x = x.next) {
if (x.item == null) {
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;
}
}
原文地址:https://www.cnblogs.com/darkclouds/p/11604958.html
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Laravel框架实现多个视图共享相同数据的方法详解
- python如何快速生成时间戳
- 从python读取sql的实例方法
- PHP+iframe模拟Ajax上传文件功能示例
- Centos7 Yum安装PHP7.2流程教程详解
- PHP session垃圾回收机制实例分析
- thinkphp5框架调用其它控制器方法 实现自定义跳转界面功能示例
- Python常用库Numpy进行矩阵运算详解
- PHP设计模式之建造者模式(Builder)原理与用法案例详解
- PHP大文件切割上传并带进度条功能示例
- PHP设计模式之观察者模式入门与应用案例详解
- Python文件夹批处理操作代码实例
- ThinkPHP框架结合Ajax实现用户名校验功能示例
- Laravel框架Eloquent ORM新增数据、自定义时间戳及批量赋值用法详解
- PHP使用HTML5 FileApi实现Ajax上传文件功能示例