双向链表详解

目录

  • 带头双向循环链表
    • 带头双向循环链表的实现
    • 带头双向循环链表的功能实现
      • 创造新节点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、算数运算符的复合运算 复合运算符是用于自己 自己进行运算…

源码解读——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;、线圈&…

MySQL 死锁案例解析一则

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

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

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

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) 转换元素 结合元素 互换元素

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

在深度学习与流体力学深度融合的背景下&#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…

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

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

SpringBootSpringCloud升级可能会出现的问题

1.背景 之前负责过我们中台的SpringBoot和Cloud的升级&#xff0c;特次记录分享一下项目中可能出现的问题&#xff0c;方便后续的人快速定位问题。以及下述选择的解决方案都是基于让升级的服务影响和改动最小以及提供通用的解决方案的提前进行选择的。 1.1版本说明 升级前&a…

陇剑杯 省赛 攻击者1 CTF wireshark 流量分析

陇剑杯 省赛 攻击者1 题目 链接&#xff1a;https://pan.baidu.com/s/1KSSXOVNPC5hu_Mf60uKM2A?pwdhaek 提取码&#xff1a;haek ├───LogAnalize │ ├───linux简单日志分析 │ │ linux-log_2.zip │ │ │ ├───misc日志分析 │ │ acce…

Vue3项目 网易严选_学习笔记

Vue3项目 网易严选_第一天 主要内容 项目搭建vuex基础路由设计首页顶部和底部布局 学习目标 知识点要求项目搭建掌握vuex基础掌握路由设计掌握首页顶部和底部布局掌握 一、项目搭建 1.1 创建项目 vue create vue-wangyi选择vue3.0版本 1.2 目录调整 大致步骤&#xff…

Workerman开启ssl方法如下

参考地址 Workerman开启ssl方法如下-遇见你与你分享 准备工作&#xff1a; 1、Workerman版本不小于3.3.7 2、PHP安装了openssl扩展 3、已经申请了证书&#xff08;pem/crt文件及key文件&#xff09;放在了/etc/nginx/conf.d/ssl下 4、配置文件 location /wss { proxy_set…

unity制作拼接地图

前段时间有个朋友问我想要制作一款地图编辑器&#xff0c;最开始我还想着在一个平面用节点切割制作地图编辑器这倒是也行&#xff0c;但不太好控制每一个点&#xff0c;如果未来项目大了&#xff0c;更加不好维护。 偶然间翻到一篇文章&#xff1a;unity地图边缘检测 或许我们…

基于数字孪生的城市建模和仿真

近年来&#xff0c;数字孪生概念几乎呈爆炸式增长&#xff0c;利用该概念的科学文章数量呈指数级增长就证明了这一点。 这一概念源自制造业&#xff0c;使用 CAD 模型可以创建组件和产品的精确数字复制品。 该术语最早的使用可以追溯到 2003 年&#xff0c;通常归功于 Grieves …

vue3第二十节(新增编译宏defineModel)

为什么会需要使用defineModel() 注意&#xff1a;defineModel() 需要在3.4及以上版本才可使用&#xff1b; 组件之间通讯&#xff0c;通过 props 和 emits 进行通讯,是单向数据流&#xff0c;比如&#xff1a;props是自上而下的&#xff08;父组件数据修改导致子组件更新&…

生成人工智能体:人类行为的交互式模拟论文与源码架构解析(2)——架构分析 - 核心思想环境搭建技术选型

4.架构分析 4.1.核心思想 超越一阶提示&#xff0c;通过增加静态知识库和信息检索方案或简单的总结方案来扩展语言模型。 将这些想法扩展到构建一个代理架构&#xff0c;该架构处理检索&#xff0c;其中过去的经验在每个时步动态更新&#xff0c;并混合与npc当前上下文和计划…

【动态规划】切割钢条详解python

1. 问题介绍和应用场景 切割钢条问题是运筹学和算法设计中的一个经典问题&#xff0c;涉及如何最优化切割有限资源以最大化收益。这个问题经常用作动态规划教学的入门案例&#xff0c;同时在工业生产中也有实际应用&#xff0c;比如在金属加工业中如何切割原材料以减少浪费并增…

EelasticSearch是什么?及EelasticSearch的安装

一、概述 Elasticsearch 是一个基于 Apache Lucene 构建的开源分布式搜索引擎和分析引擎。它专为云计算环境设计&#xff0c;提供了一个分布式的、高可用的实时分析和搜索平台。Elasticsearch 可以处理大量数据&#xff0c;并且具备横向扩展能力&#xff0c;能够通过增加更多的…