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

news2025/1/10 11:16:52

🎉🎉🎉欢迎莅临我的博客空间,我是池央,一个对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;
}

a3a432f8c86347a58ca032d1f485207d.png

2.打印

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

3.尾插

5a93f6ba31674491a04c3b29d8f41e5f.png

只有头结点时的尾插:

54509d768a324af4a387a3b1fb464425.png

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

4.头插

1948a25fe63a4d8c9e237272adf4fdff.png

// 头插
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;
}

790b4cb260344d90bacbf1549363339a.png

5.尾删

1efd723dd7134570835dfeb5ceffba02.png

// 尾删
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.头删

639c4cff529f42e8917aa85221262988.png

// 头删
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.在指定位置之后插入数据

9e9aaa6d63c345648d51464be4357ba5.png

// 指定位置之后插入
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,无法增加校验

9d31adf4092a4ca6bd0fce646f3973c7.png

// 删除指定节点
void LTErase(LTNode* pos)
{
    assert(pos);
	//pos->next pos->prev
	pos->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;
}

四、双链表的主要应用

1.双向遍历

当需要频繁地在链表中向前和向后移动时,双链表非常有用。与单链表只能从头节点开始遍历不同,双链表可以从任何节点开始向前或向后遍历。

2.实现LRU(最近最少使用)缓存

LRU缓存策略常用于操作系统、数据库和缓存系统中,用于确定哪些数据应当被移除或替换,以便为新的数据腾出空间。在LRU缓存中,最近最少使用的数据项会被移除。使用双链表可以方便地将最近访问的节点移到链表的前端,并在需要时从链表的后端移除节点。

3.实现双向队列(Deque)

双向队列是一种具有队列和栈的性质的数据结构,支持从两端插入和删除元素。使用双链表可以轻松地实现这样的数据结构。

4.撤销和重做功能

在许多文本编辑器、图形编辑器和应用程序中,用户可能需要撤销或重做之前的操作。使用双链表可以存储用户的操作历史,并允许用户向前或向后遍历这些操作。

5.文件系统的元数据管理

在文件系统中,文件和目录的元数据(如名称、大小、修改日期等)通常存储在链表中。由于文件系统需要支持删除和插入操作,并且可能需要从任意位置开始遍历,因此双链表是一个合适的选择。

6.网络协议中的数据传输

在某些网络协议中,数据需要在不同的节点之间传输,并且可能需要在传输过程中进行插入或删除操作。双链表可以方便地实现这样的数据传输机制。

7.内存管理

在某些操作系统和内存管理系统中,双链表被用于跟踪和管理内存块。例如,在内存分配和回收过程中,双链表可以用于记录哪些内存块是可用的,哪些是被占用的。

8.范围查询

在某些应用场景中,可能需要查找位于某个范围内的所有元素。使用双链表可以方便地实现这样的范围查询操作,因为可以从任意节点开始向前或向后遍历链表。

下回预告:栈

持续更新中...

敬请期待

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

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

相关文章

乡村振兴的农业品牌建设:打造农业品牌,提升农产品附加值,增强乡村经济竞争力,实现美丽乡村经济繁荣

目录 一、引言 二、农业品牌建设的重要性 (一)提升农产品附加值 (二)增强乡村经济竞争力 (三)实现美丽乡村经济繁荣 三、农业品牌建设的现状及问题 (一)现状 (二…

Go微服务: Nacos的搭建和基础API的使用

Nacos 概述 文档:https://nacos.io/docs/latest/what-is-nacos/搭建:https://nacos.io/docs/latest/quickstart/quick-start-docker/有很多种搭建方式,我们这里使用 docker 来搭建 Nacos 的搭建 这里,我们选择单机模式&#xf…

java实现图书系统源码

建包和类: Book Book: package Book;public class Book {private String name;private String author;private int price;private String type;private boolean isBorrowed;public Book(String name, String author, int price, String type) {this.name name;this.author …

【Qnx 】Qnx IPC通信PPS

Qnx IPC通信PPS Qnx自带PPS服务,PPS全称Persistent Publish/Subscribe Service,就是常见的P/S通信模式。 Qnx PPS的通信模式是异步的,Publisher和Subscriber也无需关心对方是否存在。 利用Qnx提供的PPS服务,Publisher可以通知多…

OrangePi KunPengPro | 开发板开箱测评之学习与使用

OrangePi KunPengPro | 开发板开箱测评之学习与使用 时间:2024年5月23日20:51:12 文章目录 OrangePi KunPengPro | 开发板开箱测评之学习与使用概述1.参考2.资料、工具3.使用3-1.通过串口登录系统3-2.通过SSH登录系统3-3.安装交叉编译工具链3-4.复制文件到设备3-5.第…

Android 使用 ActivityResultLauncher 申请权限

前面介绍了 Android 运行时权限。 其中,申请权限的步骤有些繁琐,需要用到:ActivityCompat.requestPermissions 函数和 onRequestPermissionsResult 回调函数,今天就借助 ActivityResultLauncher 来简化书写。 步骤1:创…

攻防世界[GoodRe]

攻防世界[GoodRe] 学到知识: 逆向的精髓:三分懂,七分蒙。TEA 算法快速识别(蒙): 数据处理的形式:进入加密时的数据和加密结束后的数据,处理时数据的分组等等,都能用来…

AtCoder Beginner Contest 355 A~F

A.Who Ate the Cake?(思维) 题意 已知有三个嫌疑人,有两个证人,每个证人可以指出其中一个嫌疑人不是罪犯,如果可以排除两个嫌疑人来确定犯人,输出犯人的身份,如果无法确定,输出"-1"。 分析 …

Pytorch(Overview)

目标 如何利用pytorch完成学习系统? 理解神经网络(neural networks)和深度学习(deep learning)基础。 需要了解线性代数和概率论数理统计等相关关系,和python编程语言。 讨论理解 到底什么是human int…

vue3项目+TypeScript前端项目 ———— elemnet-plus,svg图标配置,sass,mock数据

一.集成element-plus 官网地址 安装 pnpm install element-plus 引入 // main.ts import { createApp } from vue import ElementPlus from element-plus import element-plus/dist/index.css import App from ./App.vueconst app createApp(App)app.use(ElementPlus) app.…

esp32-idf 开发踩坑记录

现象 直接使用原始命令编译idf.py build 但是提示idf 版本错误 卸载旧版本 编译出错build 问题 然后删除编译文件后,重新编译,还是出错 解决方法1 最后发现是因为项目所在文件夹有中文目录,把项目迁移到英文目录后,重新编译&a…

重学java 46.集合 ① Collection集合

事常与人违,事总在人为 —— 24.5.26 集合 知识导航 1.集合的特点以及作用 2.使用collection接口中的方法 3.使用迭代器迭代集合 4.ArrayList以及LinkedList的使用 5.使用增强for遍历集合 一、单列集合框架的介绍 1.长度可变的容器:集合 2.集合的特点 a.…

TCP/IP协议(一)

一.报文和协议 协议有什么作用?协议定义通信实体间所交换报文的格式和次序,以及在报文发送和/或接收或者其他事件方面所采取的行动(响应)。 什么是报文?指在网络中传输的数据单元,网络通讯的基本单位。(HTTP报文、TCP报…

录屏技巧:win11怎么录屏?这5个电脑录屏方法快速了解下

无论您是想进行工作演示还是游戏直播,电脑录屏都有很大帮助。录制 Win 11 屏幕在很多方面都非常有效,因为它能让事情变得更简单。但 Win11怎么录屏呢?如果您仍有困惑,请查看本篇文章中列出的5个方法。在本文中,我们列出…

Python 获取当前IP地址(爬虫代理)

Python 获取当前IP地址(爬虫代理) 在Python中,获取当前的公网IP地址通常涉及到发送一个请求到外部服务,因为本地IP地址通常只在你的私有网络内部是可见的,而公网IP地址是由你的ISP(互联网服务提供商&#x…

猫抓(cat-catch)插件的常规用法

目录 1.1、前言1.2、抓取图片资源1.3、抓取音频资源1.4、抓取视频资源 1.1、前言 本文将介绍利用猫抓(cat-catch)插件如下抓取网页上的图片、音频、视频等资源,猫抓(cat-catch)插件的安装及设置请参考推荐一款媒体影音…

【网络技术】【Kali Linux】Wireshark嗅探(十五)SSDP(简单服务发现协议)报文捕获及分析

往期 Kali Linux 上的 Wireshark 嗅探实验见博客: 【网络技术】【Kali Linux】Wireshark嗅探(一)ping 和 ICMP 【网络技术】【Kali Linux】Wireshark嗅探(二)TCP 协议 【网络技术】【Kali Linux】Wireshark嗅探&…

picamera配opencv做发现移动物体后录像50秒

本来是想配合上一篇写的测距传感器数据打开摄像头录制个50秒实时画面,后来这个测距传感器(因为我是歪用,用来识别范围内的移动物体)给的数据,false alarming还是太高了。于是想到使用本人之前深恶痛绝的opencv来试一试…

如何使用Kimi和通义千问辅助快速阅读论文

说明 上一篇博文我介绍了最新阅读的一篇TinyML的论文。我有个习惯就是使用Google Schloar跟踪当前最新的论文,只要在Google Schloar中设置好关键字,它每天就把最新的相关论文的链接和摘要发送到邮箱里面。不过现在论文太多了,每篇都认真读取…

分享免费的手机清理软件app,一款国外开发的手机清理神器,让手机再战两年!

手机内存越来越大,软件却越来越占地方,就像微信这家伙,轻轻松松就吃了十几个G! 害得阿星8128G的手机,本来想换新的,结果用了这款Avast Cleanup软件,瞬间感觉手机还能再战两年! 注意…