双向链表详解

目录

  • 带头双向循环链表
    • 带头双向循环链表的实现
    • 带头双向循环链表的功能实现
      • 创造新节点LTNode* CreateLTNode(LTDataType x)
        • 代码
      • 初始化链表LTNode*LTInit(LTNode* phead)
        • 代码
      • 打印链表void LTPrint(LTNode* phead)
        • 代码
      • 链表尾插void LTPushBack(LTNode* phead, LTDataType x)
        • 代码
      • 链表头插 void LTPushFront(LTNode* phead, LTDataType x)
        • 代码
      • 链表尾删 void LTPopBack(LTNode* phead)
      • 链表头删void LTPopFront(LTNode* phead)
        • 代码
      • 链表查找void LTFind(LTNode* phead)
        • 代码
      • 在链表任意位置前插入void LTInsert(LTNode* pos, LTDataType x)
        • 代码
      • 在链表任意位置前删除void LTErase(LTNode* pos, LTNode* phead, LTDataType x)
        • 代码
      • 链表销毁void LTDestory(LTNode* phead)
        • 代码
  • 顺序表和链表的区别和联系
    • 链表(双向)优势
    • 顺序表问题

感谢各位大佬对我的支持,如果我的文章对你有用,欢迎点击以下链接
🐒🐒🐒 个人主页
🥸🥸🥸 C语言
🐿️🐿️🐿️ C语言例题
🐣🐣🐣 python
🐓🐓🐓 数据结构C语言

以下我写的一些文章,如果在阅读这篇文章过程中有疑惑的可以看一下
动态内存管理(下malloc free等函数的用法
动态内存管理(下)free空间等一些问题
自定义类型结构体(下)计算结构体内存大小的方法
自定义类型结构体(中)计算结构体内存大小的方法
自定义类型结构体(上)结构体的用法
C语言深入理解指针(非常详细)(一)指针的用法
C语言深入理解指针(非常详细)(二)指针的用法,以及野指针问题,和assert用法
C语言深入理解指针(非常详细)(三)二级指针
单链表详解
顺序表详解

带头双向循环链表

带头双向循环链表的结构是链表中最复杂的.一般用于单独存储数据

带头双向循环链表的结构如图
在这里插入图片描述
这个链表中的head为哨兵位,哨兵位只存储他前一个节点(尾节点)和后一个节点(头节点)的地址,不存储有效的数据,当链表没有节点的时候,哨兵位也必须存在,这种情况就是哨兵位的箭头都指向他自己
在这里插入图片描述

我们有了这个哨兵位节点后就可以轻松的实现尾插,不用像之前的单向链表一样,要遍历一遍找到尾节点后再实现尾插

带头双向循环链表的实现

typedef int LTDataType;
typedef struct ListNode
{struct ListNode* next;struct ListNode* prev;LTDataType val;
}LTNode;

next为指向后一个节点的指针,prev为指向前一个节点的指针,data为存储的有效数据

带头双向循环链表的功能实现

创造新节点LTNode* CreateLTNode(LTDataType x)

代码
LTNode* CreateLTNode(LTDataType x)
{LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));if (newnode == NULL){perror("malloc fail");exit(-1);}newnode->val = x;newnode->next = NULL;newnode->prev = NULL;return newnode;
}

初始化链表LTNodeLTInit(LTNode phead)

初始化链表是让链表只有一个哨兵位,让哨兵位的next和prev都指向他自己,因为CreateLNode必须要传入一个值,所以我们固定传一个-1进去

代码
LTNode*LTInit(LTNode* phead)
{LTNode* newnode = CreateLTNode(-1);phead->next = phead;phead->prev = phead;return phead;
}

打印链表void LTPrint(LTNode* phead)

打印链表需要注意的一点就是我们应该怎么去判断结束,因为这个链表是循环链表,他不像单向不循环链表一样,当指针指向空时就结束停止打印

所以我们需要用到循环的这个特点,当我们循环完一遍后就可以停止打印

具体过程就是,我们定义一个cur指针指向phead->next(phead是哨兵位,没有有效数据,所以直接跳过),判断结束的条件是while(cur!=phead)(循环完一遍后回到哨兵位),每次循环让cur=cur->next

代码
void LTPrint(LTNode* phead)
{assert(phead);printf("哨兵位<->");LTNode* cur = phead->next;while (cur != phead){printf("%d<->", cur->val);cur = cur->next;}
}

链表尾插void LTPushBack(LTNode* phead, LTDataType x)

在这里插入图片描述
我们需要用tail指针指向尾节点,让尾节点的next指向newnode,并且让newnode的prev指向tail
在这里插入图片描述
然后就是让新的尾节点和head连接起来,我们就让newnode的next指向head,head的prev指向newnode,就实现了尾插
在这里插入图片描述
我们还需要注意,这个链表有没有空链表的情况,换句话来说就是实现这个函数时有没有必要assert(phead)

当链表为空时,就代表着这个链表没有任何节点(包括哨兵位),哨兵位是不可以为空的,即使链表没有有效数据,哨兵位也必须存在,所以我们需要保证传进来的phead是不可以为空

代码
void LTPushBack(LTNode* phead, LTDataType x)
{assert(phead);LTNode* tail = phead->prev;LTNode* newnode = CreateLTNode(x);tail->next = newnode;newnode->next = phead;newnode->prev = tail;phead->prev = newnode;
}

链表头插 void LTPushFront(LTNode* phead, LTDataType x)

代码

方法一

void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = CreateLTNode(x);newnode->next = phead->next;phead->next->prev = newnode;phead->next = newnode;newnode->prev = phead;
}

方法二

void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = CreateLTNode(x);LTNode* first = phead->next;phead->next = newnode;newnode->prev = phead;newnode->next = first;first->prev = newnode;
}

链表尾删 void LTPopBack(LTNode* phead)

尾插需要用一个指针tailprev保存尾节点的前一个地址,删除尾节点后让tailprev->next指向phead,再让phead->prev指向tailprev

我们需要注意当链表中只有一个哨兵位时,我们是不可以删除的,所以我们需要断言提醒

void LTPopBack(LTNode* phead)
{assert(phead);LTNode* tail = phead->prev;LTNode* tailprev = tail->prev;free(tail);tailprev->next = phead;phead->prev = tailprev;
}

链表头删void LTPopFront(LTNode* phead)

链表的头删有三个指针,phead=head,first=phead->next,second=first->next,删除链表时我们需要先让phead->next指向second.然后让second->prev指向phead,最后释放掉first,让first的prev和next都指向空,
在这里插入图片描述

在这里插入图片描述
这种方法也适用于链表中只有一个节点的情况
因为second为first->next,first->next是指向哨兵位,所以second也就指向哨兵位了
在这里插入图片描述
在这里插入图片描述
当链表只有一个哨兵位时,first和second都是phead,如果我们将first对空间释放掉的话,那就意味着phead的空间也会被释放,这样就会出现野指针,为了防止这样的情况出现,我们需要加一个断言assert(phead->!=phead)

代码
void LTPopFront(LTNode* phead)
{assert(phead);assert(phead->next!=phead);LTNode* first = phead->next;LTNode* second = first->next;phead->next = second;second->prev = phead;free(first);first = NULL;
}

链表查找void LTFind(LTNode* phead)

代码
LTNode* LTFind(LTNode* phead, LTDataType x)
{assert(phead);LTNode* cur = phead->next;while (cur != phead){if (cur->val){return cur;}cur = cur->next;}return NULL;
}

在链表任意位置前插入void LTInsert(LTNode* pos, LTDataType x)

要在双链表pos位置前插入,比较简单的一种做法就是设一个指针posprev,让posprev=pos->prev,之后的过程就如下图
在这里插入图片描述
在这里插入图片描述
注意pos可以等于phead,因为是双向循环链表,所以当pos=phead时,我们可以理解成尾插
在这里插入图片描述
在这里插入图片描述

代码
void LTInsert(LTNode* pos, LTDataType x)
{assert(pos);LTNode* posPrev = pos->prev;LTNode* newnode = CreateLTNode(x);posPrev->next = newnode;newnode->prev = posPrev;newnode->next = pos;pos->prev = newnode;
}

在链表任意位置前删除void LTErase(LTNode* pos, LTNode* phead, LTDataType x)

删除的时候我们需要判断pos位置释放为哨兵位,其他的都和前面的差不多

代码
void LTErase(LTNode* pos, LTNode* phead, LTDataType x)
{assert(pos);assert(pos != phead);LTNode* posNext = pos->next;LTNode* posPrev = pos->prev;posPrev->next = posNext;posNext->prev = posPrev;free(pos);pos = NULL;
}

链表销毁void LTDestory(LTNode* phead)

链表的销毁其实传入的参数应该为LTNode**phead,包括前面的链表删除,因为如果只是一级指针,那么这种情况就和我之前写的单链表(链表的尾插void SLPushBack(SLNode** pphead, SLNDataType x))这部分相似,但是用LTNode* phead当参数也是可以的,只不过需要用完函数后,在函数外面将指针变成空

代码
void LTDestory(LTNode* phead)
{assert(phead);LTNode* cur = phead->next;while (cur != phead){LTNode* next = cur->next;free(cur);cur = next;}free(phead);phead = NULL;
}

顺序表和链表的区别和联系

链表(双向)优势

1:任意位置插入删除时间复杂度O(1)
2:按需申请释放,合理利用空间
缺点
1:下标随机访问不方便时间复杂度为O(N)(像数组一样下标访问是不方便的)

顺序表问题

1:头部或者中间插入效率低,要挪动空间,时间复杂度为O(N)
2:空间不够需要扩容,且扩容有一定的消耗,可能存在一定的空间浪费
3:只适合尾插尾删
优点
1:支持下标随机访问,时间复杂度O(1)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/824787.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C#语法知识之运算符

3、运算符 1、算数运算符 1、赋值符号 //把右侧的值赋给左侧的变量2、算数运算符 _ * / float f 1 / 2f; %3、算数运算符的优先级 //乘除余优先级高于加减 括号可以改变优先级&#xff0c;优先计算括号内的内容4、算数运算符的复合运算 复合运算符是用于自己 自己进行运算…

2024-04-17 学习笔记(李宏毅AI基础课,图像预处理)

1.Hung-yi Lee 摘要&#xff1a;这篇文章介绍了李宏毅教授的课程内容和作业安排&#xff0c;涵盖了深度学习的理论、优化、泛化、特殊网络结构、特殊训练技术、生成对抗网络(GAN)以及深度强化学习等主题。 Raiden说&#xff1a;基础知识&#xff0c;常学常新 2.软包装锂离子电…

源码解读——SplitFed: When Federated Learning Meets Split Learning

源码地址 1. 源码概述 源码里一共包含了5个py文件 单机模型&#xff08;Normal_ResNet_HAM10000.py&#xff09;联邦模型&#xff08;FL_ResNet_HAM10000.py&#xff09;本地模拟的SFLV1&#xff08;SFLV1_ResNet_HAM10000.py&#xff09;网络socket下的SFLV2&#xff08;SF…

51单片机入门_江协科技_33~34_OB记录的自学笔记_LED呼吸灯与PWM直流马达调速

33. 直流电机驱动(PWM) 33.1. 直流电机介绍 •直流电机是一种将电能转换为机械能的装置。一般的直流电机有两个电极&#xff0c;当电极正接时&#xff0c;电机正转&#xff0c;当电极反接时&#xff0c;电机反转 •直流电机主要由永磁体&#xff08;定子&#xff09;、线圈&…

笔记本触摸板的使用

使用电脑很多面年了&#xff0c;但很少使用触摸板&#xff0c;最近公司配置了台新笔记本触摸板很丝滑&#xff0c;像着改变一下多年使用电脑的习惯&#xff0c;做个笔记&#xff0c;使用它&#xff0c;适应它&#xff01; 单手指&#xff1a; 单击→左键 两次单击→打开文件夹&…

MySQL 死锁案例解析一则

原文链接&#xff1a;https://www.modb.pro/db/448666 一、问题背景某业务模块反馈数据库最近出现过几次死锁告警的情况&#xff0c;本文总结了这次死锁排查的全过程&#xff0c;并分析了导致死锁的原因及解决方案。希望给大家提供一个死锁的排查及解决思路。基础环境&#xff…

绩效考核管理:激发潜力,实现双赢

绩效考核管理是现代企业管理中不可或缺的一环&#xff0c;它不仅关乎员工的个人发展&#xff0c;更影响着企业的整体战略目标实现。本文将从绩效考核管理的意义、目标设定、考核方法、激励措施以及持续改进等方面展开论述&#xff0c;探讨如何通过有效的绩效考核管理激发员工潜…

Win10系统WSL2烧录SD卡(USB储存设备)

众做周知在嵌入式开发中经常需要制作SD卡系统来启动开发板&#xff0c;最近从虚拟机转到WSL发现不能像以前那样对SD卡进行操作了&#xff0c;记录下解决方法&#xff08;我的系统环境是Win10WSL2&#xff09; 编译WSL2内核 由于WSL2的内核默认没有添加USB存储设备的驱动的支持…

一.NODE MCU(ESP8285,ESP8286)开发环境搭建

一.序言: 1.esp8285长什么样? 2.esp8285是什么,能做什么? 通过上面图片,看到上面的芯片,是带有多个阵脚的单片机。实际上,看着该芯片很小,但是却具有完整的wifi无线蓝牙功能,它本身可以运行一个极简的linux小系统,并且该极简的小linux系统具备无线蓝牙功能。。它同…

Linux: 性能: sysctl vs echo vs直接使用fopen

简介 在实际的生产中&#xff0c;需要对系统参数做修改&#xff0c;有三种方式可以实现&#xff0c;一个是sysctl命令来修改&#xff0c;一个是使用echo 命令来写入&#xff0c;另一个是使用fopen/write接口函数来操作配置文件。 这个对比也是相当的明显&#xff0c;echo要比s…

54岁前港姐与好友因一事反目成仇,20年后方破冰

现年54岁的前「金牌司仪」陈淽菁&#xff08;前名&#xff1a;陈芷菁&#xff09;是1994年落选港姐&#xff0c;之后加入TVB参演电视剧《天地男儿》、《壹号皇庭》入屋&#xff0c;后因口齿伶俐而转战主持界。2017年陈淽菁离巢&#xff0c;外出以个人名义成立「陈芷菁工作室」&…

每日学习笔记:C++ STL算法之容器元素转换、结合、互换

本文API 转换元素 transform(sourceBeg,sourceEnd,destBeg, op) 结合元素 transform(source1Beg,source1End,source2Beg,destBeg, op) 互换元素 swap_ranges(sourceBeg,sourceEnd,destBeg) 转换元素 结合元素 互换元素

聚焦ChatGPT:让论文写作更高效更精准

ChatGPT无限次数:点击直达 html 聚焦ChatGPT&#xff1a;让论文写作更高效更精准 引言 在当今信息爆炸的时代&#xff0c;撰写高质量论文变得越发重要。然而&#xff0c;许多研究者和学者在论文写作过程中常常遇到困难&#xff0c;例如构思内容、整合观点和确保表达准确。…

什么是Cookies?请求Cookies和响应 Cookies的关系

一、什么是cookies 在早期的网络发展中&#xff0c;如何管理状态一直是一个棘手的问题。由于HTTP协议的无状态特性&#xff0c;服务器无法判断连续的两个请求是否来自同一个浏览器。为了解决这个问题&#xff0c;最初的方案是在请求时将一些参数嵌入到页面中&#xff0c;并在…

深度学习驱动的流体力学计算与应用

在深度学习与流体力学深度融合的背景下&#xff0c;科研边界不断拓展&#xff0c;创新成果层出不穷。从物理模型融合到复杂流动模拟&#xff0c;从数据驱动研究到流场智能分析&#xff0c;深度学习正以前所未有的力量重塑流体力学领域。近期在Nature和Science杂志上发表的深度学…

ARM_day8:温湿度数据采集应用

1、IIC通信过程 主机发送起始信号、主机发送8位(7位从机地址1位传送方向(0W&#xff0c;1R))、从机应答、发数据、应答、数据传输完&#xff0c;主机发送停止信号 2、起始信号和终止信号 SCL时钟线&#xff0c;SDA数据线 SCL高电平&#xff0c;SDA由高到低——起始信号 SC…

继承的初步

完成两个类&#xff0c;一个类Animal&#xff0c;表示动物类&#xff0c;有一个成员表示年龄。一个类Dog&#xff0c;继承自Animal&#xff0c;有一个新的数据成员表示颜色&#xff0c;合理设计这两个类&#xff0c;使得测试程序可以运行并得到正确的结果。 函数接口定义&…

我为什么选择做程序员

我选择做程序员的原因有多个方面。首先&#xff0c;我对计算机科学和技术有着浓厚的兴趣。从小我就对计算机的工作原理和软件开发充满好奇&#xff0c;喜欢探索新技术和解决问题。这种兴趣促使我深入学习和研究计算机领域的知识&#xff0c;最终选择了程序员这一职业。 其次&a…

php正则表达式压缩与格式化html,css,js

注意事项 只支持压缩含一个<script></script>的html,且变量内多个空格也会被压缩为一个 格式化html单标签需要添加/结束 压缩html 去除<!-- -->内的全部内容 多个空白符变为一个空格 去除> <内的空白符 压缩css 去除/* */内的全部内容 多个空白符变为一…

汽车零部件制造迎来智能化升级,3D视觉定位系统助力无人化生产线建设

随着新能源汽车市场的蓬勃发展&#xff0c;汽车零部件制造行业正面临着前所未有的机遇与挑战。为了提高产能和产品加工精度&#xff0c;某专业铝合金汽车零部件制造商决定引进智能生产线&#xff0c;其中&#xff0c;对成垛摆放的变速箱壳体进行机床上料成为关键一环。 传统的上…