单链表你别再找我了,我怕双向链表误会

news2025/1/23 3:48:30

目录

带头双向循环链表的创建和初始化

创建一个新的结点(方便复用)

链表判空

链表打印

链表尾插

链表尾删

链表头插

链表头删

任意插入

任意删除

链表查找

链表销毁

完整代码


😎前言

  • 之前我们讲了结构最简单,实现起来却最为复杂的单链表——不带头单向不循环链表,但其往往不会被运用到存储数据当中,而是作为一些数据结构的子结构(如哈希表)。
  • 而今天我们要学的则是听起来,看起来结构最为复杂,但是实施起来却最为简单的带头双向循环链表。

带头双向循环链表的创建和初始化

首先我们需要一个哨兵位,因为在单链表的学习当中我们发现,如果没有哨兵位,当链表为空的时候插入结点就要改变头结点,这时候就要传二级指针或者返回值,也要和其他情况区分,不能一套代码解决所有情况。但如果带哨兵位头节点的话,二级指针和返回值都不需要用到,并且头删头插尾插尾删都特别的方便,它的优势处,下面会一一体现。

结点的定义也和单链表不同,多了一个指针指向前一个结点

typedef struct ListNode
{
	LTDataType data;
	struct ListNode* next;
	struct ListNode* prev;
}ListNode;

 此时的head就是我们的哨兵位,但是不存储有效数据。如果链表为空,则head的prev以及next都指针都指向自己。

基于以上思路,我们可以写出创建一个哨兵位的代码了,也就是双向链表的建立。

ListNode* ListCreate()
{
	ListNode*dummmyhead=(ListNode*)malloc(sizeof(ListNode));
	dummmyhead->next = dummmyhead;
	dummmyhead->prev = dummmyhead;
	return dummmyhead;
}

创建一个新的结点(方便复用)

  • 在后续的插入接口中,要经常创建一个新的结点进行插入,所以在开头我们可以先将其制作成一个接口,方便代码的复用,这是很重要的工程项目思维。
  • 创建新的结点,其实就是向内存申请一块空间,用malloc即可
  • 防止野指针问题,每次创建的新结点的prev和next指针我们都置空。之后将这个节点的地址返回即可,因为该节点的空间是在堆上的,函数栈帧销毁不会影响这段空间。

按照以上思路,我们可以写出以下函数接口

//创建一个新节点,方便复用
ListNode* BuyNode(int x)
{
	ListNode* newNode=(ListNode*)malloc(sizeof(ListNode));
	if (newNode == NULL)
	{
		perror("malloc fail");
		return NULL;
	}

	newNode->data = x;
	newNode->next = NULL;
	newNode->prev = NULL;

	return newNode;
}

链表判空

  • 和单链表一样,当链表为空了,就不可以再进行删除操作了,因此在后面的删除接口中判空是要用到的,我们还是可以先提前将其封装成一个接口,这样可以提高代码的可读性
  • 这里需要运用到bool值,空返回true;不空返回false

上文创建头结点的时候提到了哨兵位的prev和next指针都指向自己,所以如果链表为空,那么dummyhead->next=dummyhead

根据以上思路我们可以写出以下函数接口

bool ListEmpty(ListNode* pHead)
{
	assert(pHead);

	return pHead->next == pHead;
}

链表打印

  • 打印是为了观察以后的各个接口是否正确,比起调试来说更简单直观一些
  • 将写的代码可视化了,增加了我们的一点成就感
  • 遍历除了dummyhead以外的结点,直到重新指向dummyhead的时候结束
  • 自己可以设计一些箭头,看起来更加好看一点

下面为函数接口实现:

// 双向链表打印
void ListPrint(ListNode* pHead)
{
	assert(pHead);

	ListNode* cur = pHead->next;
	printf("dummyhead<-> ");
	while (cur != pHead)
	{
		printf("%d <-> ",cur->data);
		cur = cur->next;
	}
	printf("dummyhead ");
}

链表尾插

  • 单链表中我们要从前往后找到最后一个结点,然后再插入,时间复杂度为O(N)。
  • 而双向链表实现起来十分简单,因为哨兵位的前一个结点就是链表的尾了,这样时间复杂度直接降为O(1),效率提升的让人不敢相信。
  • 如果此时链表中没有有效的节点,那么此时直接得到一个节点与头节点连接即可,不需要考虑单链表章节是否要更新头指针的情况,这也体现了带头的优处。

下面为函数接口实现:

// 双向链表尾插
void ListPushBack(ListNode* pHead, LTDataType x)
{
	assert(pHead);

	ListNode* newNode=BuyNode(x);
	ListNode* tail = pHead->prev;

	pHead->prev = newNode;
	newNode->next = pHead;
	newNode->prev = tail;
	tail->next = newNode;

}

链表尾删

  • 尾删就是要删除最后一个结点,所以我们要找到最后一个结点的前一个结点(dummyhead->prev->prev),将其与头结点相连,然后free掉尾结点。
  • 如果此时没有节点,这是判空的作用就来了,没有节点当然就是不给删咯,直接assert断言暴打。

  • 即使链表只有一个结点,也没有任何影响,大家可以自己手动画图一下,十分好理解

以下为函数实现接口:

void ListPopBack(ListNode* pHead)
{
	assert(pHead);
	assert(!ListEmpty(pHead));

	ListNode* tail = pHead->prev;
	ListNode* prev = tail->prev;
	
	pHead->prev = prev;
	prev->next = pHead;
	free(tail);
}

链表头插

  • 头插,即头结点和新结点相连,再将新结点与原来的头结点相连,这样新结点就成为了新的头结点啦!
  • 为了避免改变结点指向后写出错误代码,我们在修改指向前可以先定义几个变量,也增加代码可读性

以下为函数接口实现:

// 双向链表头插
void ListPushFront(ListNode* pHead, LTDataType x)
{
	assert(pHead);

	ListNode* head = pHead->next;
	ListNode* newNode = BuyNode(x);

	pHead->next = newNode;
	newNode->prev = pHead;
	newNode->next = head;
	head->prev = newNode;

}

链表头删

  • 头删,即将哨兵位的下一个结点删除,将哨兵位和头结点的下一个结点连接起来。
  • 依然为了防止将链表指向修改后导致的代码书写错误,我们可以先定义变量记录头结点的下一个指针
  • 如果此时没有节点,这是判空的作用就来了,没有节点当然就是不给删咯,直接assert断言暴打。
  • 即使链表只有一个结点,也没有任何影响,大家可以自己手动画图一下,十分好理解
// 双向链表头删
void ListPopFront(ListNode* pHead)
{
	assert(pHead);
	assert(!ListEmpty(pHead));

	ListNode* head = pHead->next;
	ListNode* next = head->next;
	pHead->next = next;
	next->prev = pHead;
	free(head);
}

任意插入

  • 这里的任意插入,是将你要插入的节点插入到你想要插入的位置的前面。
  • 同样的,这里需要传递一个节点的指针(pos)代表你要插入的位置
  • 既然是在想插入的位置的前面插入,那么这里就需要存放一下该位置的前一个节点的地址,然后进行连接即可

以下为相关代码的实现:

// 双向链表在pos的前面进行插入
void ListInsert(ListNode* pos, LTDataType x)
{
	assert(pos);

	ListNode* newNode = BuyNode(x);
	ListNode* prev = pos->prev;

	prev->next = newNode;
	newNode->prev = prev;
	newNode->next = pos;
	pos->prev = newNode;
}

任意删除

  • 任意删除,是删除pos位置,这个pos就有序列表是你指定要删除的那个节点的地址。

  • 既然是删除pos位置,就需要存放一下pos的前一个节点的地址和pos的下一个节点的地址,以便于连接。最后释放pos位置的节点即可。

下面为函数接口实现:

// 双向链表删除pos位置的节点
void ListErase(ListNode* pos)
{
	// 判断pos的有效性
	assert(pos);

	// 存放pos的下一个节点的地址	
	ListNode* next = pos->_next;
	// 存放pos的前一个节点的地址
	ListNode* prev = pos->_prev;
	// 删除(释放)pos
	free(pos);
	// 连接
	prev->_next = next;
	next->_prev = prev;
}

链表查找

  • 查找也就是从头结点往后一个一个看是否与值匹配,找到的话就返回首个匹配的节点位置(也可以自己修改)
  • 单链表是到NULL结束循环,而这里则是当指针重新指向dummyhead的时候,意味着所有结点都已经查找完了,此时返回NULL

下面为函数接口实现:

// 双向链表查找
ListNode* ListFind(ListNode* pHead, LTDataType x)
{
	assert(pHead);

	ListNode* cur = pHead->next;
	while (cur != pHead)
	{
		if (cur->data == x)
		return cur;
		else cur = cur->next;
	}
	return NULL;
}

链表销毁

  • 因为各个结点都是malloc出来的,所以如果不在使用后销毁可能会导致内存泄漏,所以我们要养成良好习惯,即使很多编译器已经很明智了
  • 销毁的对象包括dummyhead
  • 需要额外记录被销毁的结点的下一个结点

下面为函数接口实现:

// 双向链表销毁
void ListDestory(ListNode* pHead)
{
	assert(pHead);

	ListNode* cur = pHead->next;
	while (cur != pHead)
	{
		ListNode* next = cur->next;
		free(cur);
		cur = next;
	}
	free(pHead);
}

总结

至此,常用的接口我们已经实现的差不多了,我们会发现,双向链表正是靠其复杂的结构从而使得代码实现起来非常方便,不得不佩服想出这种结构的人。

相对于单链表,这种带头双向循环链表简直是六边形链表,任何方面都碾压了单链表。

单链表别再来找我了,我怕双向链表误会👉👈

以下是完整代码,有需求的小伙伴们可以拿走哦

完整代码

#include"List.h"

// 创建返回链表的头结点.
ListNode* ListCreate()
{
	ListNode*dummmyhead=(ListNode*)malloc(sizeof(ListNode));
	dummmyhead->next = dummmyhead;
	dummmyhead->prev = dummmyhead;
	return dummmyhead;
}
//创建一个新节点,方便复用
ListNode* BuyNode(int x)
{
	ListNode* newNode=(ListNode*)malloc(sizeof(ListNode));
	if (newNode == NULL)
	{
		perror("malloc fail");
		return NULL;
	}

	newNode->data = x;
	newNode->next = NULL;
	newNode->prev = NULL;

	return newNode;
}
// 双向链表打印
void ListPrint(ListNode* pHead)
{
	assert(pHead);

	ListNode* cur = pHead->next;
	printf("dummyhead<-> ");
	while (cur != pHead)
	{
		printf("%d <-> ",cur->data);
		cur = cur->next;
	}
	printf("dummyhead ");
}
//判断链表是否为空
bool ListEmpty(ListNode* pHead)
{
	assert(pHead);

	return pHead->next == pHead;
}
// 双向链表尾插
void ListPushBack(ListNode* pHead, LTDataType x)
{
	assert(pHead);

	ListNode* newNode=BuyNode(x);
	ListNode* tail = pHead->prev;

	pHead->prev = newNode;
	newNode->next = pHead;
	newNode->prev = tail;
	tail->next = newNode;

}
// 双向链表尾删
void ListPopBack(ListNode* pHead)
{
	assert(pHead);
	assert(!ListEmpty(pHead));

	ListNode* tail = pHead->prev;
	ListNode* prev = tail->prev;
	
	pHead->prev = prev;
	prev->next = pHead;
	free(tail);
}
// 双向链表头插
void ListPushFront(ListNode* pHead, LTDataType x)
{
	assert(pHead);

	ListNode* head = pHead->next;
	ListNode* newNode = BuyNode(x);

	pHead->next = newNode;
	newNode->prev = pHead;
	newNode->next = head;
	head->prev = newNode;

}
// 双向链表头删
void ListPopFront(ListNode* pHead)
{
	assert(pHead);
	assert(!ListEmpty(pHead));

	ListNode* head = pHead->next;
	ListNode* next = head->next;
	pHead->next = next;
	next->prev = pHead;
	free(head);
}
// 双向链表查找
ListNode* ListFind(ListNode* pHead, LTDataType x)
{
	assert(pHead);

	ListNode* cur = pHead->next;
	while (cur != pHead)
	{
		if (cur->data == x)
		return cur;
		else cur = cur->next;
	}
	return NULL;
}
// 双向链表在pos的前面进行插入
void ListInsert(ListNode* pos, LTDataType x)
{
	assert(pos);

	ListNode* newNode = BuyNode(x);
	ListNode* prev = pos->prev;

	prev->next = newNode;
	newNode->prev = prev;
	newNode->next = pos;
	pos->prev = newNode;
}
// 双向链表删除pos位置的节点
void ListErase(ListNode* pos)
{
	assert(pos);

	ListNode* prev = pos->prev;
	ListNode* next = pos->next;
	prev->next = next;
	next->prev = prev;
	free(pos);

}
// 双向链表销毁
void ListDestory(ListNode* pHead)
{
	assert(pHead);

	ListNode* cur = pHead->next;
	while (cur != pHead)
	{
		ListNode* next = cur->next;
		free(cur);
		cur = next;
	}
	free(pHead);
}

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

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

相关文章

kettle win11 启动闪退 --启动日志

一、启动闪退 思路&#xff1a; 找原因找启动日志根据启动日志查看启动失败的原因 二、找启动日志 采用debug模式启动 查看控制台–根据控制台操作 看生成的启动日志文件 查看日志 DEBUG: Using PENTAHO_JAVA_HOME DEBUG: _PENTAHO_JAVA_HOMEE:\java8 DEBUG: _PENTAHO…

记一次靶场搭建与渗透测试

渗透目标 通过Windows7打入工作组环境&#xff0c;穿透两层内网拿到DC&#xff08;域控制器&#xff09;权限 环境搭建 环境搭建 网络拓扑 虚拟机网络配置 渗透测试 永恒之蓝外网打点 nmap -sS 192.168.2.0/24扫描外网存活主机&#xff0c;发现两台主机192.168.2.128和192…

芯片电源附近为什么放置的是0.1uF电容

日常使用情况 我们在电源滤波电路上可以看到各种各样的电容&#xff0c;100uF、10uF、100nF、10nF不同的容值&#xff0c;而在我们使用中常常会在芯片电源附近放置0.1uF电容&#xff0c;以TB67S109AFNG应用手册为例&#xff0c;其中推荐使用的也是0.1uF的电容 电容的特性 数字…

分享几款小白从零开始学习的会用到的工具/网站

大二狗接触编程也有两年了&#xff0c;差生文具多这大众都认可的一句话&#xff0c;在这里蹭一下这个活动分享一下从0开始学习编程有啥好用的工具 目录 伴侣一、Snipaste截图工具 伴侣二、Postman软件&#xff08;可用ApiPost平替&#xff09; 伴侣三、字体图标网站 伴侣四…

BlockChain-Account_TakeOver

题目描述 ECDSA 签名 假设我们的私钥为 d A d_A dA​而公钥为 Q A Q_A QA​&#xff0c; Q A d A ⋅ G Q_Ad_A\cdot G QA​dA​⋅G&#xff0c;接下来就是签名的过程&#xff0c;要签名的消息为 m m m 取 e H A S H ( m ) e HASH(m) eHASH(m)取 e e e的左边的 L n L_n L…

Baumer工业相机堡盟工业相机软件CameraExplorer常见功能使用说明二

Baumer工业相机堡盟工业相机软件CameraExplorer常见功能使用说明二 Baumer工业相机Baumer工业相机CE软件图像/视频存储功能Baumer工业相机CE软件记录日志文件功能Baumer工业相机CE软件查看图像Buffer及数据流统计信息 Baumer工业相机 Baumer工业相机堡盟相机是一种高性能、高质…

汇编五、伪指令与汇编程序结构

1、伪指令 1.1、概念 (1)伪指令是用于对汇编过程进行控制的指令&#xff0c;该类指令并不是可执行指令&#xff0c;没有对应机器码&#xff0c;只用于汇编过程中为汇编程序提供汇编信息&#xff0c;帮助编译器编译。 1.2、ASM51提供的伪指令 伪指令分为如下几类。 1.2.1、…

zuul源码分析

zuul源码解析 zuul与springboot整合的依赖 <dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-netflix-zuul</artifactId></dependency>看到starter第一反应就是springboot的自动装配? 我们去…

简历制作、投递、(实习)面试技巧!!

目录 1.为什么要找实习 2.什么时候找实习 3.制作简历 4.简历注意事项 5.IT后端的校招的要求 6.简历里面写什么&#xff1f; 7.简历模板的选择 8.个人信息 9.求职意向 10.专业技能 11.项目经验 12.其他注意事项 13.找工作的手段 14.找工作的态度 ​编辑 15.面试…

IntelliJ IDEA 修改内存大小

idea有个配置文件&#xff0c;可以设置内存大小的&#xff0c;就跟咱的jvm的内存里面的堆大小&#xff0c;栈大小等等&#xff0c;可以设置的&#xff0c;而且设置了之后&#xff0c;你这个的性能就会得到提升。具体看下面怎么修改。 先说所要修改的文件 idea.vmoptions 的位置…

总结常见评价指标

整理一下在机器学习中常见的评价指标&#xff0c;包括&#xff1a; 混淆矩阵&#xff0c;TPR&#xff0c;FPR&#xff0c;TNR&#xff0c;FNR&#xff1b;Precision&#xff0c;Recall&#xff0c;Accuracy&#xff0c;F-score(F1-meature)ROC曲线&#xff0c;AUC&#xff1b; …

[Vue warn]: You may have an infinite update loop in a component render function

老板让该一个bug&#xff0c;结果一连出现好几个问题&#xff0c;然后报错也是很奇葩&#xff0c;在源代码上不能直接定位到&#xff0c;只知道在当前页面上出现的问题&#xff0c;弄了好久&#xff0c;给大家分享一下解决的经验&#xff1a; You may have an infinite update …

2023 年Windows MySql 5.7,MySql 8.0 下载安装教程, 附详细图解

文章目录 下载 MySQL 安装程序安装 MySQL 数据库安装示例数据库连接到 MySQL 服务器 在本教程中&#xff0c;我们展示如何在 Windows 平台上下载和安装 MySQL 的详细步骤。 在 Windows 平台上安装 MySQL 很简单&#xff0c;并不需要太复杂的步骤。按照本文的步骤操练起来就可以…

【Win10错误】从80190001错误码恢复

目录 一、说明 二、操作过程和错误显示 三、修复过程 四、网上的其它参考意见 一、说明 出现0x80190001错误码&#xff0c;其原因是网络认证问题引起。但不是网络断开或路由不通而引起。一般是本地身份cooki无法认证而引起&#xff0c;一般出现在登录认证过程。本篇告诉大家…

2.4G无线游戏手柄方案开发

对于游戏玩家来说&#xff0c;好的外设才能有更好的游戏体验。相比于传统的有线手柄&#xff0c;2.4G无线游戏手柄采用2.4GHz射频无线连接方式&#xff0c;摆脱了连线的困扰。相比于鼠标键盘&#xff0c;游戏手柄在大部分游戏上的使用体验都会更好&#xff0c;让你的游戏体验更…

【MATLAB第30期】基于MATLAB的adaboost多分类预测集成学习模型(四种模型GDA高斯判别分析、Knn、NB朴素贝叶斯、SVM)

【MATLAB第30期】基于MATLAB的adaboost多分类预测集成学习模型&#xff08;四种模型GDA高斯判别分析、Knn、NB朴素贝叶斯、SVM&#xff09; 一、简介 弱分类器 %1.GDA高斯判别分析 %2.Knn (NumNeighbors 5) K邻近 %3.Naive Bayes 朴素贝叶斯 %4.SVM 支持向量机 强分类器 1.a…

【Vue】生命周期

文章目录 生命周期概念一、生命周期图示二、生命周期1.beforeCreate&#xff08;&#xff09;{}2.created&#xff08;&#xff09;{}3.beforeMount&#xff08;&#xff09;{}4.mounted&#xff08;&#xff09;{}5.beforeUpdate&#xff08;&#xff09;{}6.updated&#xff…

C语言预处理详解

参考文章&#xff1a;c语言预处理 目录 程序的翻译环境和执行条件 翻译环境 编译本身也分为几个阶段 预处理 预处理指令 运行环境 程序执行的过程 预处理 预定义符号 #define #define定义标识符 #define定义宏 宏的申明方式 #define替换规则 #和## #的作用 ##…

OTP语音芯片 NV170D在充电桩的语音方案应用

新能源汽车是我国应对气候变化、推动绿色发展的战略举措&#xff0c;在政策和市场需求的推动下&#xff0c;我国新能源汽车产销量双双增长&#xff0c;新能源汽车保有量地稳步增长将会促进充电桩需求的扩大&#xff0c;企业也将进一步在电动汽车充电桩领域布局。 2022年10月11日…

Linux下的Tomcat的安装详解--值得一看

如有错误&#xff0c;敬请谅解&#xff01; 此文章仅为本人学习笔记&#xff0c;仅供参考&#xff0c;如有冒犯&#xff0c;请联系作者删除&#xff01;&#xff01; 目录 简述静态网页和动态网页的区别。 简述 Webl.0 和 Web2.0 的区别。 tomcat8的安装&#xff0c;配置服…