链表是基本的数据结构,尤其双向链表在应用中最为常见,LinkedList 就实现了双向链表。今天我们一起手写一个双向链表。

文中涉及的代码可访问 GitHub:https://github.com/UniqueDong/algorithms.git

上次我们说了「单向链表」的代码实现,今天带大家一起玩下双向链表,双向链表的节点比单项多了一个指针引用 「prev」。双向链表就像渣男,跟「前女友」和「现女友」,还有一个「备胎』都保持联系。前女友就像是前驱节点,现女友就是 「当前 data」,而「next」指针就像是他套住的备胎。每个 Node 节点有三个属性,类比就是 「前女友」+ 「现女友」 + 「备胎」。

使用这样的数据结构就能实现「进可攻退可守」灵活状态。

接下来让我们一起实现『渣男双向链表』。

定义Node

节点分别保存现女友、前女友、跟备胎的联系方式,这样就能够实现一三五轮换运动(往前看有前女友,往后看有备胎),通过不同指针变可以找到前女友跟备胎。就像渣男拥有她们的联系方式。


  1. private static class Node<E> {
  2. //现女友
  3. E item;
  4. // 备胎
  5. Node<E> next;
  6. // 前女友
  7. Node<E> prev;
  8. public Node(Node<E> prev, E item, Node<E> next) {
  9. this.prev = prev;
  10. this.item = item;
  11. this.next = next;
  12. }
  13. }

代码实现

定义好渣男节点后,就开始实现我们的双向链表。类似过来就是一个渣男联盟排成一列。我们还需要定义两个指针分别指向头结点和尾节点。一个带头大哥,一个收尾小弟。

  1. public class DoubleLinkedList<E> extends AbstractList<E> implements Queue<E> {
  2. transient int size = 0;
  3. /**
  4. * Pointer to first node.
  5. * Invariant: (first == null && last == null) ||
  6. * (first.prev == null && first.item != null)
  7. */
  8. transient Node<E> first;
  9. /**
  10. * Pointer to last node.
  11. * Invariant: (first == null && last == null) ||
  12. * (last.next == null && last.item != null)
  13. */
  14. transient Node<E> last;
  15. }

头节点添加

新的渣男进群了,把他设置成群主带头大哥。首先构建新节点,prev = null,带头大哥业务繁忙,不找前女友,所以 prev = null;next 则指向原先的 first。

  1. 如果链表是空的,则还要把尾节点也指向新创建的节点。
  2. 若果链表已近有数据,则把原先 first.prev = newNode。
  1. @Override
  2. public void addFirst(E e) {
  3. linkFirst(e);
  4. }
  5. /**
  6. * 头结点添加数据
  7. *
  8. * @param e 数据
  9. */
  10. private void linkFirst(E e) {
  11. final Node<E> f = this.first;
  12. Node<E> newNode = new Node<>(null, e, f);
  13. // first 指向新节点
  14. first = newNode;
  15. if (Objects.isNull(f)) {
  16. // 链表是空的
  17. last = newNode;
  18. } else {
  19. // 将原 first.prev = newNode
  20. f.prev = newNode;
  21. }
  22. size++;
  23. }

尾节点添加

将新进来的成员放在尾巴。

第一步构建新节点,把 last 指向新节点。

第二步判断 last 节点是否是空,为空则说明当前链表是空,还要把 first 指向新节点。否则就需要把原 last.next 的指针指向新节点。

  1. @Override
  2. public boolean add(E e) {
  3. addLast(e);
  4. return true;
  5. }
  6. private void addLast(E e) {
  7. final Node<E> l = this.last;
  8. Node<E> newNode = new Node<>(l, e, null);
  9. last = newNode;
  10. if (Objects.isNull(l)) {
  11. // 链表为空的情况下,设置 first 指向新节点
  12. first = newNode;
  13. } else {
  14. // 原 last 节点的 next 指向新节点
  15. l.next = newNode;
  16. }
  17. size++;
  18. }

指定位置添加

分为两种情况,一个是在最后的节点新加一个。一种是在指定节点的前面插入新节点。

在后面添加前面尾巴添加已经说过,对于在指定节点的前面插入需要我们先找到指定位置节点,然后改变他们的 prev next 指向。


  1. @Override
  2. public void add(int index, E element) {
  3. checkPositionIndex(index);
  4. if (index == size) {
  5. linkLast(element);
  6. } else {
  7. linkBefore(element, node(index));
  8. }
  9. }
  10. /**
  11. * Links e as last element.
  12. */
  13. void linkLast(E element) {
  14. addLast(element);
  15. }
  16. /**
  17. * Inserts element e before non-null Node succ.
  18. */
  19. private void linkBefore(E element, Node<E> succ) {
  20. // assert succ != null
  21. final Node<E> prev = succ.prev;
  22. // 构造新节点
  23. final Node<E> newNode = new Node<>(prev, element, succ);
  24. succ.prev = newNode;
  25. if (Objects.isNull(prev)) {
  26. first = newNode;
  27. } else {
  28. prev.next = newNode;
  29. }
  30. size++;
  31. }

节点查找

为了优化,根据 index 查找的时候先判断 index 落在前半部分还是后半部分。前半部分通过 first 开始查找,否则通过 last 指针从后往前遍历。

  1. @Override
  2. public E get(int index) {
  3. checkElementIndex(index);
  4. return node(index).item;
  5. }
  6. /**
  7. * Returns the (non-null) Node at the specified element index.
  8. */
  9. Node<E> node(int index) {
  10. // 优化查找,判断 index 在前半部分还是后半部分。
  11. if (index < (this.size >> 2)) {
  12. // 前半部分,从头结点开始查找
  13. Node<E> x = this.first;
  14. for (int i = 0; i < index; i++) {
  15. x = x.next;
  16. }
  17. return x;
  18. } else {
  19. // 后半部分,从尾节点开始查找
  20. Node<E> x = this.last;
  21. for (int i = size - 1; i > index; i--) {
  22. x = x.prev;
  23. }
  24. return x;
  25. }
  26. }

查找 Object 所在位置 indexOf ,若找不到返回 -1

  1. @Override
  2. public int indexOf(Object o) {
  3. int index = 0;
  4. if (Objects.isNull(o)) {
  5. for (Node<E> x = first; x != null; x = x.next) {
  6. if (x.item == null) {
  7. return index;
  8. }
  9. index++;
  10. }
  11. } else {
  12. for (Node<E> x = first; x != null; x = x.next) {
  13. if (x.item.equals(o)) {
  14. return index;
  15. }
  16. index++;
  17. }
  18. }
  19. return -1;
  20. }

判断 链表中是否存在 指定对象 contains ,其实还是利用 上面的 indexOf 方法,当返回值 不等于 -1 则说明包含该对象。


  1. @Override
  2. public boolean contains(Object o) {
  3. return indexOf(o) != -1;
  4. }

节点删除

有两种删除情况:

  1. 根据下标删除指定位置的节点。
  2. 删除指定数据的节点。

删除指定位置节点

  1. 首先判断该 index 是否合法存在。
  2. 查找要删除的节点位置,重新设置被删除节点关联的指针指向。

node() 方法已经在前面的查找中封装好这里可以直接调用,我们再实现 unlink 方法,该方法还会用于删除指定对象,所以这抽出来实现复用。也是最核心最不好理解的方法,我们多思考画图理解下。

  1. @Override
  2. public E remove(int index) {
  3. checkElementIndex(index);
  4. return unlink(node(index));
  5. }
  6. public final void checkElementIndex(int index) {
  7. if (!isElementIndex(index))
  8. throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
  9. }
  10. /**
  11. * Tells if the argument is the index of an existing element.
  12. */
  13. private boolean isElementIndex(int index) {
  14. return index >= 0 && index < size();
  15. }
  16. /**
  17. * Unlinks non-null node x.
  18. */
  19. private E unlink(Node<E> x) {
  20. // assert x != null;
  21. final E element = x.item;
  22. final Node<E> next = x.next;
  23. final Node<E> prev = x.prev;
  24. // 若 只有一个节点,那么会执行 prev == null 和 next == null 分支代码
  25. // 若 prev == null 则说明删除的是头结点,主要负责 x 节点跟前驱节点的引用处理
  26. if (Objects.isNull(prev)) {
  27. first = next;
  28. } else {
  29. prev.next = next;
  30. x.prev = null;
  31. }
  32. // 若 next 为空,说明删除的是尾节点,主要负责 x 与 next 节点 引用的处理
  33. if (Objects.isNull(next)) {
  34. last = prev;
  35. } else {
  36. next.prev = prev;
  37. x.next = null;
  38. }
  39. x.item = null;
  40. size--;
  41. return element;
  42. }

分别找出被删除节点 x 的前驱和后继节点,要考虑当前链表只有一个节点的情况,最后还要把被删除节点的 的 next 指针 ,item 设置 null,便于垃圾回收,防止内存泄漏。

删除指定数据

这里判断下数据是否是 null , 从头节点开始遍历链表,当找到索要删除的节点的时候低啊用前面封装好的 unlink 方法实现删除。


  1. @Override
  2. public boolean remove(Object o) {
  3. if (Objects.isNull(o)) {
  4. for (Node<E> x = first; x != null; x = x.next) {
  5. if (x.item == null) {
  6. unlink(x);
  7. return true;
  8. }
  9. }
  10. } else {
  11. for (Node<E> x = first; x != null; x = x.next) {
  12. if (o.equals(x.item)) {
  13. unlink(x);
  14. return true;
  15. }
  16. }
  17. }
  18. return false;
  19. }

完整代码可以参考 GitHub:https://github.com/UniqueDong/algorithms.git

加群跟我们一起探讨,欢迎关注 MageByte,我第一时间解答。

推荐阅读

1.跨越数据结构与算法

2.时间复杂度与空间复杂度

3.最好、最坏、平均、均摊时间复杂度

4.线性表之数组

5.链表导论-心法篇

6.单向链表正确实现方式

原创不易,觉得有用希望读者随手「在看」「收藏」「转发」三连。

7L-双线链表实现的更多相关文章

  1. JAVA 基本数据结构--数组、链表、ArrayList、Linkedlist、hashmap、hashtab等

    概要 线性表是一种线性结构,它是具有相同类型的n(n≥0)个数据元素组成的有限序列.本章先介绍线性表的几个基本组成部分:数组.单向链表.双向链表:随后给出双向链表的C.C++和Java三种语言的实现. ...

  2. MySQL记录之间是单向链表还是双向链表?

    前言 本文的观点是基于MySQL使用Innodb存储引擎的情况下进行的! 很多渠道说:MySQL数据按照主键大小依次排列,记录之间是双向链表连起来.如果说我告诉你这种说法很大程度上是错的,你肯定说我在 ...

  3. Java 集合系列05之 LinkedList详细介绍(源码解析)和使用示例

    概要  前面,我们已经学习了ArrayList,并了解了fail-fast机制.这一章我们接着学习List的实现类——LinkedList.和学习ArrayList一样,接下来呢,我们先对Linked ...

  4. Java集合系列:-----------05LinkedList的底层实现

    前面,我们已经学习了ArrayList,并了解了fail-fast机制.这一章我们接着学习List的实现类--LinkedList.和学习ArrayList一样,接下来呢,我们先对LinkedList ...

  5. Java 集合系列 04 LinkedList详细介绍(源码解析)和使用示例

    java 集合系列目录: Java 集合系列 01 总体框架 Java 集合系列 02 Collection架构 Java 集合系列 03 ArrayList详细介绍(源码解析)和使用示例 Java ...

  6. LinkedList源代码深入剖析

    第1部分 LinkedList介绍LinkedList简介 public class LinkedList<E> extends AbstractSequentialList<E&g ...

  7. LinkedHashMap源码分析及实现LRU

    概述 从名字上看LinkedHashMap相比于HashMap,显然多了链表的实现.从功能上看,LinkedHashMap有序,HashMap无序.这里的顺序指的是添加顺序或者访问顺序. 基本使用 @ ...

  8. 【由浅入深理解java集合】(三)——集合 List

    第一篇文章中介绍了List集合的一些通用知识.本篇文章将集中介绍List集合相比Collection接口增加的一些重要功能以及List集合的两个重要子类ArrayList及LinkedList. 一. ...

  9. LinkedList源码分析和实例应用

    1. LinkedList介绍 LinkedList是继承于AbstractSequentialList抽象类,它也可以被当作堆栈.队列或者双端队列使用. LinkedList实现了Deque接口,即 ...

  10. java集合系列之LinkList

    概要  第1部分 LinkedList介绍第2部分 LinkedList数据结构第3部分 LinkedList源码解析(基于JDK1.6.0_45) 第5部分 LinkedList示例 转载请注明出处 ...

随机推荐

  1. 前端BOM和DOM

      前端基础之BOM和DOM   前戏 到目前为止,我们已经学过了JavaScript的一些简单的语法.但是这些简单的语法,并没有和浏览器有任何交互. 也就是我们还不能制作一些我们经常看到的网页的一些 ...

  2. Web Scraper 高级用法——利用正则表达式筛选文本信息 | 简易数据分析 17

    这是简易数据分析系列的第 17 篇文章. 学习了这么多课,我想大家已经发现了,web scraper 主要是用来爬取文本信息的. 在爬取的过程中,我们经常会遇到一个问题:网页上的数据比较脏,我们只需要 ...

  3. Checkbox 勾上 不让勾下 同步手动刷新复选框状态 iview

    <Checkbox v-show="!disabledForm" ref="youwubianhuaRef" :value="youwubian ...

  4. Mybatis(一)Mybatis相关概念

    1.1 传统的JDBC实现 public static void main(String[] args) { Connection connetion = null; PreparedStatemen ...

  5. 在Linux环境安装redis步骤,且设置开机自动启动redis

    最近在linux环境安装了redis学习,目前已经安装成功且设置开机即启动状态,我把步骤流程记录了下来,分享给需要的小伙伴. 1.我在/usr/local/localsoftware/目录下创建了一个 ...

  6. 数据结构和算法:Python实现二分查找(Binary_search)

    在一个列表当中我们可以进行线性查找也可以进行二分查找,即通过不同的方法找到我们想要的数字,线性查找即按照数字从列表里一个一个从左向右查找,找到之后程序停下.而二分查找的效率往往会比线性查找更高. 一. ...

  7. c# winform 访问WebServices (通过Http方式)

    第一步.编写WebServices服务方法 [WebMethod] public void PostJson(string str, string bb) { Dictionary<string ...

  8. gdb中的gef插件

    地址 https://github.com/hugsy/gef # via the install script #下载 `gef.sh` 并执行 wget -q -O- https://github ...

  9. 机器学习 - 命名实体识别之Hidden Markov Modelling

    概述 命名实体识别在NLP的应用中也是非常广泛的,尤其是是information extraction的领域.Named Entity Recognition(NER) 的应用中,最常用的一种算法模型 ...

  10. (原)Non-local Neural Networks

    转载请注明出处: 论文: https://arxiv.org/abs/1711.07971 第三方pytorch代码: https://github.com/AlexHex7/Non-local_py ...