【数据结构】第四讲:双向链表

个人主页:深情秋刀鱼@-CSDN博客

数据结构专栏:数据结构与算法

        循环链表这个轮回的思想很有意思。它强调了不管你今生是贫是富,如果持续行善积德,下辈子就会好过,反之就会遭到报应。

        就像每个人的人生一样,欲收获就得付出代价。双向链表既然是比单链表多了如可以反向遍历查找的数据结构,那么也就要付出一些小的代价。

一、链表的分类

        链表的结构复杂多样,总计有如下八种形式。

        虽然有这么多的链表的结构,但是我们实际中最常⽤还是两种结构: 单链表 双向带头循环链表。
1. ⽆头单向⾮循环链表:结构简单,⼀般不会单独⽤来存数据。实际中更多是作为其他数据结 构的⼦结构,如哈希桶、图的邻接表等等。另外这种结构在笔试⾯试中出现很多。
2. 带头双向循环链表:结构最复杂,⼀般⽤在单独存储数据。实际中使⽤的链表数据结构,都 是带头双向循环链表。另外这个结构虽然结构复杂,但是使⽤代码实现以        后会发现结构会带 来很多优势,实现反⽽简单了。

       

二、双向链表的结构及实现

1.带头双向链表的结构

         对于带头和不带头,在学习单链表是我们将其理解为链表的头节点(链表的第一个节点),这种称呼很不严谨,在本节中的带头和不带头指的是在该链表中书否含有哨兵结点,哨兵结点在双向链表中是名副其实的头节点,哨兵节点内不存储任何有效的数据,他的唯一作用是防止在遍历链表时进入死循环。

//双向链表的定义
typedef int LTDataType;//定义双向链表的节点
typedef struct ListNode
{LTDataType data;//数据域struct ListNode* next;//指向后继节点的指针struct ListNode* prev;//指向前驱节点的指针
}ListNode;

2.创建节点

ListNode* LTbuyNode(LTDataType x)
{ListNode* newNode = (ListNode*)malloc(sizeof(ListNode));if (newNode == NULL){perror("malloc fail!");exit(1);}newNode->data = x;newNode->next = newNode;newNode->prev = newNode;//创建的新节点指向自身return newNode;
}

3.初始化

//初始化
void LTinit(ListNode** pphead)
{*pphead = LTbuyNode(-1);//哨兵位(不存储有效值)
}

4.尾插

//尾插
void LTpushBack(ListNode* phead, LTDataType x)
{assert(phead);ListNode* node = LTbuyNode(x);node->prev = phead->prev;node->next = phead;phead->prev->next = node;phead->prev = node;
}

5.打印

//打印
void LTprint(ListNode* phead)
{assert(phead);ListNode* pcur = phead->next;while (pcur != phead){printf("%d->", pcur->data);pcur = pcur->next;}printf("head\n");
}

6.头插

//头插
void LTpushFront(ListNode* phead, LTDataType x)
{assert(phead);ListNode* node = LTbuyNode(x);node->prev = phead;node->next = phead->next;phead->next->prev = node;phead->next = node;
}

7.尾删

//尾删
void LTpopBack(ListNode* phead)
{assert(phead && phead->next != phead);//链表不为空ListNode* pcur = phead->prev;phead->prev->prev->next = phead;phead->prev = phead->prev->prev;free(pcur);pcur = NULL;
}

8.头删

//头删
void LTpopFront(ListNode* phead)
{assert(phead && phead->next != phead);ListNode* pcur = phead->next;phead->next->next->prev = phead;phead->next = phead->next->next;free(pcur);pcur = NULL;
}

9.在pos位置之后插入数据

//pos位置之后插入数据
void LTinsert(ListNode* pos, LTDataType x)
{assert(pos);ListNode* node = LTbuyNode(x);pos->next->prev = node;node->prev = pos;node->next = pos->next;pos->next = node;
}

10.删除pos节点

//删除pos节点
void LTerase(ListNode* pos)
{assert(pos);pos->prev->next = pos->next;pos->next->prev = pos->prev;free(pos);pos = NULL;
}

11.查找

//查找
ListNode* LTfind(ListNode* phead, LTDataType x)
{assert(phead && phead->next != phead);ListNode* pcur = phead->next;while (pcur != phead){if (pcur->data == x){printf("找到了!\n");return pcur;}pcur = pcur->next;}printf("没有找到!\n");return NULL;
}

12.销毁

//销毁
void LTdestroy(ListNode* phead)
{assert(phead);ListNode* pcur = phead->next;while (pcur != phead){ListNode* next = pcur->next;free(pcur);pcur = next;}free(phead);phead = NULL;
}

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

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

相关文章

android 分区存储(沙盒存储)适配总结

目录 一、分区存储概念 1.外部存储分类 2.分区存储如何影响文件访问 二、分区适配方案 1. 应用分区存储的文件访问规定 (1).应用专属目录--私有目录 (2).共享目录文件--公有目录 2.MediaStore API介绍 3.Storage Access Framework介绍 三、所有文件访问权限 四、总结…

Docker容器---Harbor私有仓库部署与管理

一、搭建本地私有仓库 1、下载registry镜像 [rootlocalhost ~]#docker pull registry Using default tag: latest latest: Pulling from library/registry 79e9f2f55bf5: Pull complete 0d96da54f60b: Pull complete 5b27040df4a2: Pull complete e2ead8259a04: Pull comp…

第十五届蓝桥杯Java软件开发大学B组自我经验小结

自我介绍 23届大一 双非 计院科班 软件工程 江苏人在吉林上大学 Java蒟蒻 在学校的宣传下 有幸参加了第十五届蓝桥杯Java大学b组省赛 蓝桥杯说明 就是一个算法比赛吧 考试时间9.00到1.00 四小时 带准考证和身份证和笔 草稿纸会发 赛制是IOC就是不会给任何反馈 就是你…

【6D位姿估计】GDR-Net 单目几何引导的直接回归模型

前沿 本文介绍6D位姿估计的直接回归方法GDR-Net,它从单个RGB图像中确定物体在三维空间中的位置和方向。 它是一个端到端模型,与传统的间接方法不同,GDR-Net可以通过反向传播完全训练,简化了训练过程。 论文地址:GDR…

【C++航海王:追寻罗杰的编程之路】C++11(四)

目录 1 -> 相关文章 【C航海王:追寻罗杰的编程之路】C11(一) 【C航海王:追寻罗杰的编程之路】C11(二) 【C航海王:追寻罗杰的编程之路】C11(三) 2 -> lambda表达式 2.1 -> C98中的一个例子 2.2 -> lambda表达式 2.3 ->…

【19】JAVASE-多线程专题【从零开始学JAVA】

Java零基础系列课程-JavaSE基础篇 Lecture:波哥 Java 是第一大编程语言和开发平台。它有助于企业降低成本、缩短开发周期、推动创新以及改善应用服务。如今全球有数百万开发人员运行着超过 51 亿个 Java 虚拟机,Java 仍是企业和开发人员的首选开发平台。…

qt嵌入并控制外部程序

一、流程 1、调用Window接口模拟鼠标,键盘事件 POINT point; LPPOINT lpppoint &point; GetCursorPos(lpppoint);//获取鼠标位置 SetCursorPos(point.x, point.y);//设置鼠标位置//鼠标左键按下 mouse_event(MOUSEEVENTF_LEFTDOWN | MOUSEEVENTF_LEFTUP, poi…

Windows php 安装 Memcached扩展、php缺失 Memcached扩展、Class ‘Memcached‘ not found

在Windows系统下如何安装 php Memcached 扩展 下载dll文件 pecl地址:https://pecl.php.net/package/memcached 根据版本进行选择 : 解压下载的文件后得到了这么样的文件结构: 配置 移动dll文件到相应文件位置 重点: libme…

React的useEffect

概念:useEffect是一个React Hook函数,组件渲染之后执行的函数 参数1是一个函数,可以把它叫做副作用函数,在函数内部可以放置要执行的操作参数2是一个数组(可选参),在数组里放置依赖项&#x…

【自动化测试】使用MeterSphere进行接口测试

一、接口介绍二、接口测试的过程三、接口自动化测试执行自动化流程 四、接口之间的协议HTTP协议 五、 接口测试用例设计接口文档 六、使用MeterSphere创建接口测试创建接口定义设计接口测试用例 一、接口介绍 自动化测试按对象分为:单元测试、接口测试、UI测试等。…

拌合楼管理系统(十八)如何从一个winForm中的事件修改另外一个Form的控件的值

前言: 上篇讲述了如何手工调用海康的车牌识别摄像头进行拍照和识别车牌,我车牌识别的程序在应用的一个窗体,需要去更新另外一个窗体里面的lable中的内容为识别的车牌信息,同时还要写入到另外窗体的datagridview中。 一、实现效果 …

Java面试八股之Java中数组有没有length()方法?String呢?为什么?

Java中数组有没有length()方法?String呢?为什么? 数组: 数组没有名为length()的方法,但有一个属性叫做length。可以通过数组名直接访问这个属性来获取数组的长度(即元素个数)。这是一个整数值&…