【探索数据结构】线性表之双链表

🎉🎉🎉欢迎莅临我的博客空间,我是池央,一个对C++和数据结构怀有无限热忱的探索者。🙌

🌸🌸🌸这里是我分享C/C++编程、数据结构应用的乐园✨

🎈🎈🎈期待与你一同在编程的海洋中遨游,探索未知的技术奥秘💞

📝专栏指路:

📘【C++】专栏:深入解析C++的奥秘,分享编程技巧与实践。

📘【数据结构】专栏:探索数据结构的魅力,助你提升编程能力。

前言

之前我们已经探索了顺序表和单链表我们继续一起来探索逻辑结构里面的线性结构。线性表在逻辑结构上是连续的,线性表中双链表(本篇主角)在物理结构上是不连续的。

文章重点介绍:带头双向循环链表

一、双链表

1.概念

双链表,也叫双向链表,是链表的一种特殊形式。在双链表中,每个数据节点都有两个指针,一个指向前一个节点(前驱节点),另一个指向后一个节点(后继节点)。这种结构使得从双链表中的任意一个节点开始,都可以很方便地访问它的前驱节点和后继节点。

2.分类

(1)按不同属性分

  • 带头节点的双链表:这种双链表在第一个数据节点之前有一个头结点。头结点是为了操作的统一和方便而设立的,放在第一元素的结点之前,其数据域一般无意义(也可存放链表的长度)。有了头结点,对在第一元素结点前插入结点和删除第一结点,其操作与其它结点的操作就统一了。
  • 不带头节点的双链表:与带头结点的双链表相反,这种链表没有头结点,直接从第一个数据节点开始。

(2)按循环性分

  • 双向循环链表:在双向链表的基础上,将头结点的后驱指针指向尾节点,尾节点的前驱指针指向头结点,从而形成一个双向环
  • 双向非循环链表:这是标准的双向链表,没有形成一个环,只是简单地通过前驱和后继指针连接各个节点。

3.链表结构

typedef int LTDataType;
typedef struct LTNode LTNode;
struct LTNode
{LTDataType data;//数据LTNode* prev;//前驱指针LTNode* next;//后继指针
};

二、对双链表的操作

0.创建节点

//创建节点
LTNode* LTBuyNode(LTDataType x)
{LTNode* node = (LTNode*)malloc(sizeof(LTNode));if (node == NULL){perror("malloc fail!");exit(1);}node->data = x;node->next = node->prev = node;//不能置为空,都要指向自己return node;
}

1.初始化

后续对链表的操作都是不需要改变头结点的,哨兵位节点不能被删除,节点的地址,也不能发生改变只需传一级指针。为了保持接口的一致性。我们没有在初始化方法中选择传二级指针的方式实现

// 初始化
LTNode* LTInit()
{LTNode* phead = LTBuyNode(-1);return phead;
}

2.打印

// 打印
void LTPrint(LTNode* phead)
{//从第一个有效节点遍历LTNode* pcur = phead->next;while (pcur != phead){printf("%d->", pcur->data);pcur = pcur->next;}printf("\n");
}

3.尾插

只有头结点时的尾插:

// 尾插
void LTPushBack(LTNode* phead, LTDataType x)
{assert(phead);//phead phead->prev phead->nextLTNode* newnode = LTBuyNode(x);//先改变新节点的指针指向不影响原链表newnode->prev = phead->prev;newnode->next = phead;//不可以调换下面两句顺序,否则会找不到原链表的尾结点phead->prev->next = newnode;phead->prev = newnode;
}

4.头插

// 头插
void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);//phead phead->next LTNode* newnode = LTBuyNode(x);newnode->next = phead->next;newnode->prev = phead;//尽量不要调换下面两句顺序phead->next->prev = newnode;phead->next = newnode;
}

5.尾删

// 尾删
void LTDelBack(LTNode* phead)
{assert(phead&&phead->next);//链表不能为空//phead  phead->prev//把要删除的节点先存起来以防找不到他的前一个节点LTNode* del = phead->prev;del->prev->next = phead;phead->prev = del->prev;free(del);del = NULL;
}

6.头删

// 头删
void LTDelFront(LTNode* phead)
{assert(phead && phead->next);//链表不能为空//把要删除的节点先存起来以防找不到他的后一个节点LTNode* del = phead->next;del->next->prev = phead;phead->next = del->next;free(del);del = NULL;
}

7.查找

LTNode* LTFind(LTNode* phead, LTDataType x)
{assert(phead);//从第一个有效节点遍历LTNode* pcur = phead->next;while (pcur != phead){if (pcur->data == x){return pcur;}pcur = pcur->next;}return NULL;
}

8.在指定位置之后插入数据

// 指定位置之后插入
void LTPushPos(LTNode* pos, LTDataType x)
{assert(pos);//pos->next pos LTNode* newnode = LTBuyNode(x);//先改变新节点的指针指向不影响原链表newnode->prev = pos;newnode->next = pos->next;pos->next->prev = newnode;pos->next = newnode;
}

9.删除指定节点

pos理论上来说不能为phead,但是没有参数phead,无法增加校验

// 删除指定节点
void LTErase(LTNode* pos)
{assert(pos);//pos->next pos->prevpos->prev->next = pos->next;pos->next->prev = pos->prev;free(pos);pos = NULL;
}

10.销毁链表

// 销毁
void LTDestroy(LTNode* phead)
{assert(phead);//从第一个有效节点遍历LTNode* pcur = phead->next;while (pcur != phead){//先把要删除节点的下一个节点存起来//不然要删除后续节点无法被找到LTNode* next = pcur->next;free(pcur);pcur = next;}//把哨兵位置为空free(phead);phead = NULL;
}

三、测试(仅供参考)

#include"List.h"
void test01()
{LTNode* plist = LTInit();//尾插LTPushBack(plist, 3);LTPushBack(plist, 2);LTPushBack(plist, 1);LTPrint(plist);//头插LTPushFront(plist, 4);LTPushFront(plist, 5);LTPushFront(plist, 6);LTPrint(plist);//尾删LTDelBack(plist);LTPrint(plist);//头删LTDelFront(plist);LTPrint(plist);LTNode* find = LTFind(plist, 5);if (find == NULL){printf("没有找到\n");}else{printf("找到了\n");}/*LTErase(find);LTPrint(plist);*/LTPushPos(find, 10);LTPrint(plist);LTDestroy(plist);//为保持接口一致性没有传二级指针//需要手动把实参置为空plist = NULL;
}
int main()
{test01();return 0;
}

下回预告:栈

持续更新中...

敬请期待

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

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

相关文章

MT3041 多项式变换求值

注意点: 1.使用单调栈 2.用ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);避免超时 3.此题除了ans最好不要用long long,如果a[]和b[]都是long long 类型,可能会超内存 4.ans (ans % p p) % p;防止负数 5.使用秦九韶算法计算指数…

【Linux】语言级文件接口与系统级文件接口

目录 前言 一、回顾C语言中的文件操作 二、认识文件缓冲区 三、Linux系统提供的文件接口 四、文件描述符fd简介 Linux下C语言文件接口简单模拟实现 前言 每个编程语言都有自己的文件操作方法,在不同的操作系统下相同的语言有相同的文件操作方法。这是如何实现…

声音转文本(免费工具)

声音转文本:解锁语音技术的无限可能 在当今这个数字化时代,信息的传递方式正以前所未有的速度进化。从手动输入到触控操作,再到如今的语音交互,技术的发展让沟通变得更加自然与高效。声音转文本(Speech-to-Text, STT&…

git拉取项目前需要操作哪些?

1.输入 $ ssh-keygen -t rsa -C "秘钥说明" 按enter键 2.出现 ssh/id_rsa:(输入也可以不输入也可以) 然后按enter键 3.出现empty for no passphrase:(输入也可以不输入也可以) 然后按enter键 4.出现same passphrase again: (输入也可以不输入也…

异常检测 | PCA马氏距离异常值检测(Matlab)

异常检测 | PCA马氏距离异常值检测(Matlab) 目录 异常检测 | PCA马氏距离异常值检测(Matlab)效果一览基本介绍程序设计参考资料 效果一览 基本介绍 Matlab Pca 马氏距离异常值检测,剔除异常样本,置信椭圆可…

15:00面试,15:08就出来了,问的问题有点变态。。。

从小厂出来,没想到在另一家公司又寄了。 到这家公司开始上班,加班是每天必不可少的,看在钱给的比较多的份上,就不太计较了。没想到8月一纸通知,所有人不准加班,加班费不仅没有了,薪资还要降40%…

4. C++入门:内联函数、auto关键字、范围for及nullptr

内联函数 概念 以inline修饰的函数叫做内联函数,编译时C编译器会在调用内联函数的地方展开,没有函数调用建立栈帧的开销,内联函数提升程序运行的效率 对比C的宏 C语言不足:宏 #define ADD(x, y) ((x)(y))int main() {int ret…

OpenFeign高级用法:缓存、QueryMap、MatrixVariable、CollectionFormat优雅地远程调用

码到三十五 : 个人主页 微服务架构中,服务之间的通信变得尤为关键。OpenFeign,一个声明式的Web服务客户端,使得REST API的调用变得更加简单和优雅。OpenFeign集成了Ribbon和Hystrix,具有负载均衡和容错的能力&#xff…

期权策略交易怎么做?怎么选择期权策略?

今天期权懂带你了解期权策略交易怎么做?怎么选择期权策略?期权交易是一种金融衍生品交易方式,它给予购买者在未来特定时间内以特定价格购买(或出售)标的资产的权利。 期权策略交易怎么做? 配对看跌期权&am…

vue+springboot实现echarts数据图统计

①vue项目修改配置 安装依赖: npm i echarts -S 修改路由index.js: import Vue from vue import VueRouter from vue-router import Manager from ../views/Manager.vue // 解决导航栏或者底部导航tabBar中的vue-router在3.0版本以上频繁点击菜单报错…

00.OpenLayers快速开始

00OpenLayers快速开始 官方文档: 快速开始:https://openlayers.org/doc/quickstart.html 需要node环境 一、设置新项目 npm create ol-app my-app cd my-app npm start第一个命令将创建一个名为 my-app​ 的目录(如果您愿意,…

Python筑基之旅-MySQL数据库(一)

目录 一、MySQL数据库 1、简介 2、优点 2-1、开源和免费 2-2、高性能 2-3、可扩展性 2-4、易用性 2-5、灵活性 2-6、安全性和稳定性 2-7、丰富的功能 2-8、结合其他工具和服务 2-9、良好的兼容性和移植性 3、缺点 3-1、对大数据的支持有限 3-2、缺乏全文…

前端面试:项目细节重难点问题分享

面试官提问:我现在给你出一个项目实际遇到的问题:由于后端比较忙,所以我们这边的列表数据排序需要前端最近实现,那你会怎么实现排序呢? 答:我的回答:确实,数据都是由后端实现的&…

正点原子[第二期]Linux之ARM(MX6U)裸机篇学习笔记-19讲 串口实验UART

前言: 本文是根据哔哩哔哩网站上“正点原子[第二期]Linux之ARM(MX6U)裸机篇”视频的学习笔记,在这里会记录下正点原子 I.MX6ULL 开发板的配套视频教程所作的实验和学习笔记内容。本文大量引用了正点原子教学视频和链接中的内容。…

嵌入式硬件中PCB走线与过孔的电流承载能力分析

简介 使用FR4敷铜板PCBA上各个器件之间的电气连接是通过其各层敷着的铜箔走线和过孔来实现的。 由于不同产品、不同模块电流大小不同,为实现各个功能,设计人员需要知道所设计的走线和过孔能否承载相应的电流,以实现产品的功能,防止过流时产品烧毁。 文中介绍设计和测试FR4敷…

Windows系统安装OpenSSH使用VScode远程连接内网Linux服务器开发

文章目录 💡推荐 前言1、安装OpenSSH2、VS Code配置ssh3. 局域网测试连接远程服务器4. 公网远程连接4.1 ubuntu安装cpolar内网穿透4.2 创建隧道映射4.3 测试公网远程连接 5. 配置固定TCP端口地址5.1 保留一个固定TCP端口地址5.2 配置固定TCP端口地址5.3 测试固定公网…

Nginx/阿里云/二级域名的配置和使用

阿里云域名解析配置如下: nginx配置如下: 访问地址: zhadmin.iotzzh.com image.png

Hotcoin Research | 市场洞察:2024年5月13日-5月19日

加密货币市场表现 目前,加密货币总市值为1.32万亿,BTC占比54.41%。 本周行情呈现震荡上行的态势,BTC在5月15日-16日,有一波大的拉升,周末为震荡行情。BTC现价为67125美元。 上涨的主要原因:美国4月CPI为3…

深度学习之基于YoloV5钢材微小缺陷检测系统

欢迎大家点赞、收藏、关注、评论啦 ,由于篇幅有限,只展示了部分核心代码。 文章目录 一项目简介 二、功能三、系统四. 总结 一项目简介 一、项目背景与目标 在钢材生产过程中,由于各种因素,钢材表面可能会出现微小缺陷&#xff…

C# OpenCvSharp 模拟生成车辆运行视频

C# OpenCvSharp 模拟生成车辆运行视频 目录 效果 项目 代码 下载 效果 项目 代码 using OpenCvSharp; using OpenCvSharp.Extensions; using System; using System.Diagnostics; using System.Drawing; using System.Windows.Forms; namespace OpenCvSharp_Demo { p…