数据结构与算法—一文多图搞懂双链表

简介: 前面讲过线性表中顺序表和链表的实现和性质。但是在数据结构与算法中,双向链表无论在考察还是运用中都占有很大的比例,笔者旨在通过本文与读者一起学习分享双链表相关知识。

前言



前面讲过线性表中顺序表和链表实现和性质。但是在数据结构与算法中,双向链表无论在考察还是运用中都占有很大的比例,笔者旨在通过本文与读者一起学习分享双链表相关知识。


20190809000824266.png


双链表介绍



与单链表区别


逻辑上没有区别。他们均是完成线性表的内容。主要的区别是结构上的构造有所区别。


对于单链表:


20190809123655554.png


对于双链表:

  • 对于一个节点,有些和单链表一样有存储数据的data,指向后方的next(指针)。它拥有单链表的所有操作和内容。但是他还有一个前驱节点pre(指针)。


20190809124617909.png


结构的设计


  • 对于双链表的结构,上图也很清楚的。以前设计的单链表是带头节点的。带头节点可以方面首位的插入和删除。而这次我们抱着学习的态度搞清链表故该双链表是不带头节点的.


  • 同时,以前的单链表是不带尾节点的,这次我们带上尾节点tail。这样我们就接触了几乎所有类型啦!遇到啥也不怕了。


所以我们构造的这个双链表的的性质:


  • 不带头节点、带尾指针(tail)、双向链表。

对于node节点:


class node<T> {
    T data;
  node<T> pre;
  node<T> next;
  public node() {
  }
  public node(T data) {
    this.data = data;
  }
}


对于链表:


public class doubleList<T> {
    private node<T> head;// 头节点
  private node<T> tail;// 尾节点
  private int length;
    //各种方法  
}


具体方法的解析



  • 其实对于一个链表主要的操作还是增删。增闪的话都需要考虑是否带头节点。头插尾插中间插。并且还要考虑其中的一些细节处理。指针的运算。防止链表崩掉。因为这些操作如果不当往往会对链表的结构和证悟性带来致命的打击。而像查找那些逻辑稍微简单。也很容易排查错误。


初始化


  • 我们知道一个双链表在最初的时候它的数据肯定是为null的。那么对于这个不带头节点的双链表而言。它的head始终指向第一个真实有效的数据。tail也是如此。那么在最初没数据的时候当然要head=null,并且tail=head。(tail和head需要在一个链上)。


public doubleList() {
  head = null;
  tail = head;
  length = 0;
  }


增加


空表插入:


  • 对于空链表来说。增加第一个元素可以特殊考虑。因为在链表为空的时候headtail均为null。但head和tail又需要实实在在指向链表中的真实数据(带头指针就不需要考虑)。所以这时候就新建一个nodehead、tail等于它


node<T> teamNode = new node(data);
if (isEmpty()) {
  head = teamNode;
  tail = teamNode;  
}


头插入:


对于头插入来说。步骤很简单,只需考虑head节点的变化。

  1. 新建插入节点node
  2. head前驱指向node
  3. node后驱指向head
  4. head指向node。(这时候head只是表示第二个节点,而head需要表示第一个节点故重新赋值)


20190809182441541.gif


尾插入:


对于尾插入来说。只需考虑尾节点tail节点的变化。

  1. 新建插入节点node
  2. node前驱指向tail
  3. tail后驱指向node
  4. tail指向node。(这时候tail只是表示倒数第二个节点,而tail需要表示最后节点故重新赋值等于node即可)


20190809183120201.gif


编号插入:


对于编号插入来说。要考虑查找和插入两部,而插入既和head无关也和tail无关。

  1. 新建插入节点node
  2. 找到欲插入node的前一个节点pre。和后一个节点after
  3. node后驱指向after,after前驱指向node(次时node和后面节点的关联已经完成,但是和前面处理分离状态)


20190809185533364.png


pre后驱指向node。node前驱指向pre(此时完毕)


20190809190410151.png


整个流程的动态图为:


20190809184426414.gif


删除


单节点删除:


无论头删还是尾删,遇到单节点删除的需要将链表从新初始化!


if (length == 1)// 只有一个元素
{
  head = null;
  tail = head;
  length--;
}


头删除:


头删除需要注意的就是删除不为空时候头删除只和head节点有关

大致分为:


  1. head节点的后驱节点前驱节点改为null。(head后面本指向head但是要删除第一个先让后面那个和head断绝关系)
  2. head节点指向head.next.(这样head就指向我们需要的第一个节点了。如果有需要处理内存的语言就可以把第一个被孤立的节点删除了)


20190810002139574.png


尾删除:


尾删除需要注意的就是删除不为空时候尾删除只和tail节点有关。记得在普通链表中,我们删除尾节点需要找到尾节点的前驱节点。需要遍历整个表。而双向链表可以直接从尾节点遍历到前面。


删除的时tail所在位置的点。也就是tail所在节点要断绝和双链表的关系。

  1. tail.pre.next=null尾节点的前一个节点(pre)的后驱节点等于null
  2. tail=tail.pre尾节点指向它的前驱节点,此时尾节点由于步骤1next已经为null。完成删除


20190810102240135.gif


普通删除:


普通删除需要重点掌握,因为前两个删除都是普通删除的一个特例而已。(普通删除要确保不是头删除和尾删除)

  1. 找打将删除节点的前驱节点team(team.next是要删除的节点)
  2. team.next.next.pre=team.(欲被删除节点的后一个节点的前驱指向team,双向链表需要处理pre和next。这步处理了pre)


2019081010534338.png


  1. team.next=team.next.next;此时team.next也跳过被删除节点。


20190810111339985.png


  1. 完成删除


20190810111520598.png


整个流程的动态图为:


20190810112617911.gif


代码与测试



代码:


package LinerList;
/*
 * 不带头节点的
 */
public class doubleList<T> {
class node<T> {
  T data;
  node<T> pre;
  node<T> next;
  public node() {
  }
  public node(T data) {
    this.data = data;
  }
}
private node<T> head;// 头节点
private node<T> tail;// 尾节点
private int length;
public doubleList() {
  head = null;
  tail = head;
  length = 0;
}
boolean isEmpty() {
  return length == 0 ? true : false;
}
void addfirst(T data) {
  node<T> teamNode = new node(data);
  if (isEmpty()) {
    head = teamNode;
    tail = teamNode;
  } else {
    teamNode.next = head;
    head = teamNode;
  }
  length++;
}
void add(T data)// 尾节点插入
{
  node<T> teamNode = new node(data);
  if (isEmpty()) {
    head = teamNode;
    tail = teamNode;
  } else {
    tail.next = teamNode;
    teamNode.pre=tail;
    tail = teamNode;
  }
  length++;
}
   int length()
   {
    return length;
   }
T getElum(int index)//为了简单统一从头找
{
  node<T> team=head;
  for(int i=0;i<index;i++)//不带头节点  遍历次数-1
  {
    team=team.next;
  }
  return team.data;
}
void add(int index, T data)// 编号插入
{
  if (index == 0) {
    addfirst(data);
  } else if (index == length) {
    add(data);
  } else {// 重头戏
    node teampre = head;// 为插入的前qu
    for (int i = 0; i < index -1; i++)// 无头节点,index-1位置找到前驱节点
    {
      teampre = teampre.next;
    }
    node<T> team = new node(data);// a c 中插入B 找打a
    team.next = teampre.next;// B.next=c
    teampre.next.pre = team;// c.pre=B
    team.pre = teampre;// 关联a B
    teampre.next = team;
    length++;
  }
}
void deletefirst()// 头部删除
{
  if (length == 1)// 只有一个元素
  {
    head = null;
    tail = head;
    length--;
  } else {
    head = head.next;
    length--;
  }
}
 void deletelast() {
  if(length==1)
  {
    head=null;
    tail=head;
    length--;
  }
  else {
    tail.pre.next=null;
    tail=tail.pre;
    length--;
  }
}
 void delete(int index)
 {
   if(index==0)deletefirst();
   else if (index==length-1) {
    deletelast();
  }
   else {//删除 为了理解统一从头找那个节点  
    node<T>team=head;
    for(int i=0;i<index-1;i++)
    {
      team=team.next;
    }
    //team 此时为要删除的前节点  a  c   插入B  a为team
    team.next.next.pre=team;//c的前驱变成a
    team.next=team.next.next;//a的后驱变成c
    length--;
  }
 }
   void set(int index,T data)
   {
    node<T>team=head;
  for(int i=0;i<index-1;i++)
  {
    team=team.next;
  }
  team.data=data;
   }
@Override
public String toString() {
  node<T> team = head;
  String vaString = "";
  while (team != null) {
    vaString += team.data + " ";
    team = team.next;
  }
  return vaString;
}
}


测试:


package LinerList;
public class test {
  public static void main(String[] args) throws Exception {
// TODO Auto-generated method stub
  System.out.println("线性表测试:");
  doubleList<Integer> list = new doubleList<Integer>();
  list.add(66);
  list.addfirst(55);
  list.add(1, 101);
  list.add(-22);
  list.add(555);
  list.addfirst(9999);
  System.out.println(list.toString() + " lenth " + list.length());// 9999 55 101 66 -22 555
  // System.out.println(list.getElum(0)+" "+list.getElum(2)+" "+list.getElum(4));
  list.deletefirst();
  System.out.println(list.toString() + " lenth " + list.length());// 55 101 66 -22 555 lenth 5
  list.delete(1);
  System.out.println(list.toString() + " length " + list.length());// 55 66 -22 555 length 4
  list.delete(1);
  System.out.println(list.toString() + " length " + list.length());// 55 -22 555 length 3
  list.deletelast();
  System.out.println(list.toString() + " lenth " + list.length());// 55 -22 lenth 2
  list.deletelast();
  System.out.println(list.toString() + " lenth " + list.length());// 55 lenth 1
  list.deletelast();
  System.out.println(list.toString() + " lenth " + list.length());// lenth 0
  System.err.println("欢迎关注公众号:bigsai");
  }
}


结果图


20190810113723466.png


总结与感悟



插入、删除顺序问题


  • 很多人其实不清楚插入、删除正确的顺序是什么。其实这点没有必然的顺序,要根据题意所给的条件完成相同的结果即可!
  • 还有就是你可能会搞不清一堆next.next这些问题。这时候建议你画个图。你也可以先建一个节点,用变量名完成操作,可能会更容易一些。比如删除操作,你找到pre节点(删除前的节点)。你可以node delete=pre.next,node next=delete.next。这样你直接操作pre。delete。next三个节点会更简单。
  • 但是很多题目只给你一个node。你这时候要分析next(pre)。改变顺序。因为只有一个节点,你改变next(pre)很可能导致你遍历不到那个节点。所以这种情况要好好思考(可以参考笔者的代码实现)。
  • 至于有些语言需要删除内存的。别忘记删除。(java大法好)


其他操作问题:


  • 对于其他操作,相比增删要容易理解,可以参考代码理解。
  • 双向链表可以对很多操作进行优化。这里只是突出实现并没有写的太多。比如查找时候可以根据长度判断这个链表从头查找还是从尾查找


另外,代码写的可能不是太好,链表也没考虑线程安全问题。算法效率可能不太优。如果有什么改进或者漏洞还请大佬指出!


最后(last but not least):


喜欢的感觉可以的烦请大家动动小手关注一下把,关注后回复: 数据结构 有精心准备的系列。个人公众号交流:bigsai

欢迎交友!


目录
相关文章
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
68 4
|
3月前
|
算法 索引
❤️算法笔记❤️-(每日一刷-141、环形链表)
❤️算法笔记❤️-(每日一刷-141、环形链表)
61 0
|
9天前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】单链表的基本运算(头歌实践教学平台习题)【合集】
本内容介绍了单链表的基本运算任务,涵盖线性表的基本概念、初始化、销毁、判定是否为空表、求长度、输出、求元素值、按元素值查找、插入和删除数据元素等操作。通过C++代码示例详细解释了顺序表和链表的实现方法,并提供了测试说明、通 - **任务描述**:实现单链表的基本运算。 - **相关知识**:包括线性表的概念、初始化、销毁、判断空表、求长度、输出、求元素值、查找、插入和删除等操作。 - **测试说明**:平台会对你编写的代码进行测试,提供测试输入和预期输出。 - **通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了测试通过后的预期输出结果。 开始你的任务吧,祝你成功!
28 5
|
23天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
86 5
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
137 4
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
存储 Web App开发 算法
2024重生之回溯数据结构与算法系列学习之单双链表【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构之单双链表按位、值查找;[前后]插入;删除指定节点;求表长、静态链表等代码及具体思路详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
60 0
|
3月前
|
存储 缓存 算法
经典算法之链表篇(三)
经典算法之链表篇(三)