【数据结构与算法】 - 双向链表 - 详细实现思路及代码

news2025/1/11 12:41:05

目录
一、概述
二、双向链表
三、双向链表实现步骤
 📌3.1 C语言定义双向链表结点
 📌3.2 双向链表初始化
 📌3.3 双向链表插入数据
 📌3.4 双向链表删除数据
 📌3.5 双向链表查找数据
 📌3.6 双向链表的销毁
四、双向链表链表完整代码


在这里插入图片描述

一、概述

前几篇文章介绍了怎样去实现单链表、单循环链表,这篇文章主要介绍双向链表以及实现双向链表的步骤,最后提供我自己根据理解实现双向链表的C语言代码。跟着后面实现思路看下去,应该可以看懂代码,看懂代码后,就对双向链表有了比较抽象的理解了,最后自己再动手写一个双向链表,就基本理解这个东西了。
在这里插入图片描述

在这里插入图片描述

二、双向链表

双向链表:在单链表的每个结点中,再设置一个指向其前驱结点的指针域。
下图是 单链表
在这里插入图片描述

下图是 双向链表
在这里插入图片描述

双向链表的特点:

  1. 双向链表可以反向访问到链表的结点,因为它有指向前一个结点的指针prior
  2. 带有头结点的双向链表,为空链表时,头结点的两个指针域都指向NULL
    在这里插入图片描述
  3. 带有头结点的双向链表,为非空链表时,
    头结点的前驱指针域指向NULL,后驱指针域指向第一个结点;
    最后一个结点的前驱指针域指向前一个结点,后驱指针域指向NULL
    其他结点的前驱指针域指向前一个结点,后驱指针域指向后一个结点;
    在这里插入图片描述

在这里插入图片描述

三、双向链表实现步骤

从上面知道了双向链表的相关概念和一些特点,接下来开始实现双向链表,这里使用带有头结点的双向链表进行讲解,从初始化双向链表、插入数据、删除数据、查找数据、销毁双向链表5个操作进行说明,需要注意的是,双向链表的插入、删除操作需要改变两个指针域;其他操作基本和单链表一致。

📌3.1 C语言定义双向链表结点

为了和前几篇文章的链表做比较,双向链表结构体也尽量定义相似的。

typedef int ElemType;
typedef struct _DoubleListNode
{
	ElemType data;
	struct _DoubleListNode *prior;	// 前驱指针
	struct _DoubleListNode *next;	// 后驱指针
}DoubleListNode;
typedef DoubleListNode* DoubleLinkList;

📌3.2 双向链表初始化

因为带有头结点,初始化时就需要分配一个头结点的内存空间,且头指针会一直指向头结点。
双向链表初始化算法思路如下:

1、分配一个结点的存储空间作为头结点,并将头指针指向头结点;
2、让头结点的 prior指针 和 next指针 都指向NULL,头结点的数据填一个无效值;
3、将头指针返回给函数调用者。

C语言实现代码如下:

DoubleLinkList ListInit()
{
	DoubleLinkList list = (DoubleLinkList)malloc(sizeof(DoubleListNode));
	list->prior = NULL;
	list->next = NULL;
	list->data = -1;
	return list;
}

在这里插入图片描述

📌3.3 双向链表插入数据

双向链表插入数据大致分为两个步骤:首先,找到插入位置n的前一个结点;其次,是插入新结点,可以:先连接新结点、再指向新结点的顺序。
先连接新结点:是先把新结点的两个指针域分别连接当前结点和下个结点,new->prior = cur;new->next = cur->next;
再指向新结点:将当前节点的的指针域指向新节点,与旧节点断开,cur->next->prior = new;cur->next = new;
在这里插入图片描述

双向链表在第n个位置插入数据的算法思路:

1、定义一个结点指针cur指向头结点,用来遍历链表;
2、定义一个变量cur_i,用来表示当前结点的序号,初始化为0表示当前指向头结点;
3、将cur指针不断往后移动,直到下个位置就是插入位置n,即当cur_i==(n-1)跳出循环;
4、若结束循环后是当前结点无效,说明链表长度不够;
5、否则,说明当前结点cur的下个位置就是插入位置n,分配存储空间给新结点new;
6、把值填进新节点的数据域,用新结点prior指向当前结点,next指向当前节点的下个节点;
7、再将下个结点的prior指向新结点,当前结点的next指向新结点,完成插入操作。

C语言实现代码如下:

int ListInsert(DoubleLinkList list, int data, int n)// 将node插入到第n位,n从1开始
{
	if(list==NULL || n<1) // 判断参数有效性
		return -1;
		
	DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点
		int cur_i=0;			// cur_i表示当前结点的序号,0-头结点
	while(cur && cur_i<(n-1))// 当前结点有效,且不是插入位置的前一个结点,就后移一个
	{
		cur = cur->next;
		cur_i++;
	}
	if(!cur)			// 当前结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;	// 链表没有 n 那么长
	}
	DoubleListNode* new = (DoubleListNode*)malloc(sizeof(DoubleListNode));
	new->data = data;
	new->prior = cur;
	new->next = cur->next;
	if(cur->next)		// 在最后一个结点插入时,cur->next==NULL
		cur->next->prior = new;
	cur->next = new;
	
	return 0;
}

📌3.4 双向链表删除数据

双向链表删除结点也是需要改变两个指针域,大致步骤如下,首先,找到删除位置n的前一个结点;其次,“把前一个结点的next指针域指向删除结点del的下个结点”,“再把下个结点的prior指针域指向删除结点del的前个结点”,这样就删除了下一个结点。
在这里插入图片描述

双向链表删除第n个数据的算法思路:

1、定义一个结点指针cur指向头结点,用来遍历链表;
2、定义一个变量cur_i,用来表示下个结点的序号,初始化为0表示当前指向头结点;
3、将cur指针不断往后移动,直到下个位置就是删除位置n,即当cur_i==(n-1)跳出循环;
4、若结束循环后是最后一个结点(cur->next==NULL),说明链表长度不够;
5、否则,说明下个结点(cur->next)就是删除位置n的结点delete,赋值delete = cur->next;
6、将前一个结点的next指针域指向 del 的下个结点 ,delete->prior->next = delete->next;
7、将下一个结点的prior指针域指向 del 的前个结点 ,delete->next->prior = delete->prior;;
8、最后释放delete结点的内存,完成删除操作。

C语言实现代码如下,删除结点更关注的是下个结点(cur->next)的有效性:

// 删除第n个结点,且将删除的值通过data传出
int ListDelete(DoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点
	int cur_i=0;				// cur_i表示当前结点的序号,0-头结点
	while(cur->next && cur_i<(n-1))
	{// 下个结点有效,且当前位置不是删除位置的前一个,就后移一个
		cur = cur->next;
		cur_i++;
	}
	if(!cur->next)		// 下个结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;		// 链表没有 n 那么长
	}
	DoubleListNode *delete = cur->next;
	delete->prior->next = delete->next;
	delete->next->prior = delete->prior;
	free(delete);
	return 0;
}

📌3.5 双向链表查找数据

查找数据时,将指针指向第一个结点而非头结点,下面函数中list是头指针,指向头结点,双向链表非空时,list->next就是第一个结点;双向链表为空时,list->next == NULL。双向链表 和 单链表 查找数据的算法是一样的。

双向链表查找第n个数据的算法思路:

1、定义一个结点指针cur指向第一个结点(list->next),用来遍历链表;
2、定义一个变量cur_i,用来表示当前结点的序号,初始化为1(第一步指向的就是第一个结点);
3、若当前结点有效,且当前位置不是查找位置n,就继续后移,直到最后结点或cur_i==n跳出循环;
4、若结束循环后,当前结点无效,说明已经移动到最后,链表长度不够;
5、否则,说明当前结点(cur)就是查找位置n的结点;返回结点数据*data = cur->data。

C语言实现代码如下:

int ListFind(DoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	
	DoubleListNode* cur = list->next;// 指向第一个节点
	int cur_i=1;			// i表示当前结点的序号
	while(cur && cur_i<n)	// 当前结点有效,且当前位置不是查找位置n,就往后移动一个
	{
		cur = cur->next;
		cur_i++;
	}
	if(!cur)			// 当前结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;	// 链表没有 n 那么长
	}
	*data = cur->data;
	printf("[%s %d]find No.%d = %d\n", __FUNCTION__,__LINE__, n,*data);
	return 0;
}

📌3.6 双向链表的销毁

双向链表销毁的算法思路:

1、定义一个结点指针cur指向第一个结点,用来遍历链表;
2、定义一个结点指针next,保存下个结点地址;
3、当前指针不是指向最后一个结点的指针域就后移,进入循环:
	3.1、先保存下个结点地址,因为下个结点本来保存在cur->next,直接free(cur)会丢掉下个结点;
	3.2、删除当前结点,释放内存
	3.3、将当前指针指向前面保存好的下个结点。
4、结束循环后,已经删除完所有节点,此时需要将头结点的两个指针域都指向NULL,表示空链表。

C语言实现代码如下:

void ListDestroy(DoubleLinkList list)
{
	DoubleListNode* cur = list->next;	// 指向第一个节点
	DoubleListNode* next = NULL;		// 用于保存下个结点地址
	while(cur)	// 当前结点有效,就往后移动
	{
		next = cur->next;		// 保存下个结点地址
		//printf("[%s %d]delete %d\n", __FUNCTION__,__LINE__, cur->data);
		free(cur);				// 删除当前结点、并释放内存
		cur = next;				// 将当前结点指针指向下个结点
	}
	list->prior = NULL;
	list->next = NULL;
}

在这里插入图片描述

四、双向链表完整代码

代码只是为了更好地了解循环链表,实现过程可能存在不足,有发现的,欢迎指正,谢谢!!!
代码已在Ubuntu编译通过,可执行。

// DoubleList.c
#include <stdio.h>
#include <stdlib.h>

typedef int ElemType;
typedef struct _DoubleListNode
{
	ElemType data;
	struct _DoubleListNode *prior;	// 前驱指针
	struct _DoubleListNode *next;	// 后驱指针
}DoubleListNode;
typedef DoubleListNode* DoubleLinkList;

DoubleLinkList ListInit()
{
	DoubleLinkList list = (DoubleLinkList)malloc(sizeof(DoubleListNode));
	list->prior = NULL;
	list->next = NULL;
	list->data = -1;
	return list;
}

int ListInsert(DoubleLinkList list, int data, int n)// 将node插入到第n位,n从1开始
{
	if(list==NULL || n<1) // 判断参数有效性
		return -1;
		
	DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点
	int cur_i=0;				// cur_i表示当前结点的序号,0-头结点
	while(cur && cur_i<(n-1))// 当前结点有效,且不是插入位置的前一个结点,就后移一个
	{
		cur = cur->next;
		cur_i++;
	}
	if(!cur)			// 当前结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;	// 链表没有 n 那么长
	}
	DoubleListNode* new = (DoubleListNode*)malloc(sizeof(DoubleListNode));
	new->data = data;
	new->prior = cur;
	new->next = cur->next;
	if(cur->next)		// 在最后一个结点插入时,cur->next==NULL
		cur->next->prior = new;
	cur->next = new;
	
	return 0;
}

// 删除第n个结点,且将删除的值通过data传出
int ListDelete(DoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点
	int cur_i=0;				// cur_i表示当前结点的序号,0-头结点
	while(cur->next && cur_i<(n-1))
	{// 下个结点有效,且当前位置不是删除位置的前一个,就后移一个
		cur = cur->next;
		cur_i++;
	}
	if(!cur->next)		// 下个结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;		// 链表没有 n 那么长
	}
	DoubleListNode *delete = cur->next;
	delete->prior->next = delete->next;
	delete->next->prior = delete->prior;
	free(delete);
	return 0;
}

int ListFind(DoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	
	DoubleListNode* cur = list->next;// 指向第一个节点
	int cur_i=1;			// i表示当前结点的序号
	while(cur && cur_i<n)	// 当前结点有效,且当前位置不是查找位置n,就往后移动一个
	{
		cur = cur->next;
		cur_i++;
	}
	if(!cur)			// 当前结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;	// 链表没有 n 那么长
	}
	*data = cur->data;
	printf("[%s %d]find No.%d = %d\n", __FUNCTION__,__LINE__, n,*data);
	return 0;
}

void ListDestroy(DoubleLinkList list)
{
	DoubleListNode* cur = list->next;	// 指向第一个节点
	DoubleListNode* next = NULL;		// 用于保存下个结点地址
	while(cur)	// 当前结点有效,就往后移动
	{
		next = cur->next;		// 保存下个结点地址
		//printf("[%s %d]delete %d\n", __FUNCTION__,__LINE__, cur->data);
		free(cur);				// 删除当前结点、并释放内存
		cur = next;				// 将当前结点指针指向下个结点
	}
	list->prior = NULL;
	list->next = NULL;
}

void ListPrintf(DoubleLinkList list)
{
	DoubleListNode* cur = list->next;// 指向第一个节点
	printf("list:[");
	while(cur)
	{
		printf("%d,",cur->data);
		cur = cur->next;
	}
	printf("]\n");
}

int main()
{
	DoubleLinkList list=ListInit();
	int data=0;
	
	printf("Linklist is empty !!! \n");
	ListInsert(list, 2, 2);		// 空链表时,验证插入
	ListDelete(list, &data, 1);	// 空链表时,验证删除
	ListFind(list, &data, 1);	// 空链表时,验证查询
	ListDestroy(list);			// 空链表时,验证销毁
	
	printf("\ninsert 3 data\n");
	// 正常插入3个数据
	ListInsert(list, 1, 1);
	ListInsert(list, 2, 2);
	ListInsert(list, 3, 3);
	ListPrintf(list);
	
	printf("\n验证错误值\n");
	ListInsert(list, 5, 5);		// 验证插入
	ListDelete(list, &data, 4);	// 验证删除
	ListFind(list, &data, 4);	// 验证查询
	
	printf("\n正常操作\n");
	// 正常操作
	ListFind(list, &data, 2);
	printf("delete 2,now\n");
	ListDelete(list, &data, 2);
	ListPrintf(list);
	
	printf("Insert 4 to 2,now\n");
	ListInsert(list, 4, 2);
	ListPrintf(list);
	
	printf("Destroy ,now\n");
	ListDestroy(list);
	ListPrintf(list);

	return 0;
}

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

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

相关文章

总线仿真与测试工具CANoe介绍(图文并茂)

1、什么是CANoe CANoe是德国Vector公司的一款用于开发、测试和分析单个ECU和整个ECU网络的综合性工具,包括软件和硬件。它在整个开发过程中为网络设计者、开发和测试工程师提供支持:从规划到系统级测试。由于其多种变体和功能能够对不同的项目提供支持,被全球OEM和供应商广…

USB 速度识别

文章目录USB 速度识别低速设备全速设备高速设备空闲状态低速 / 全速设备高速设备SE0SE1USB 速度识别 在 USB host 或者 hub 的每个下行端口&#xff08;downstream facing port&#xff09;的 D 和 D- 上分别接了一个 15KΩ 的下拉电阻到地&#xff0c;这样&#xff0c;当端口…

SpringBoot自动装配原理(附面试快速答法)

文章目录SpringBoot自动装配原理1. 从调用SpringApplication构造器方法开始2. 解析启动类4.按需装配4.1 分析dubbo自动装配5. 如果定义自己的starter6. 面试答法SpringBoot自动装配原理 之前面试被问到这个题目&#xff0c;只会答一些spi、AutoConfigration注解、Import之类的&…

《SQLi-Labs》01. Less 1~5

Less-1 ~ Less-5 前言Less-1知识点题解 Less-2题解 Less-3题解 Less-4题解 Less-5知识点题解 sqli。开启新坑。 前言 对于新手&#xff0c;为了更加直观的看到 sql 注入语句&#xff0c;可以在以下文件添加两句&#xff1a; echo $sql; # 将构造的 sql 语句进行输出 echo &qu…

Maven依赖冲突分析和解决

使用maven管理jar包依赖时&#xff0c;可能会出现jar包版本冲突&#xff0c;不同版本的api调用方式可能不同&#xff0c;会出现NoSuchMethodError和ClassNotFoundException问题&#xff0c;甚至编译不通过&#xff0c;如&#xff1a;在common-lang3 的3.8.1版本中MethodUtils::…

Python中的@cache巧妙用法

缓存是一种空间换时间的策略&#xff0c;缓存的设置可以提高计算机系统的性能&#xff0c;这篇文章主要介绍了Python中的cache巧妙用法,需要的朋友可以参考下 Python中的cache有什么妙用&#xff1f; 缓存是一种空间换时间的策略&#xff0c;缓存的设置可以提高计算机系统的性…

Spark 简介与原理

目录标题1 Spark 简介与原理1.1 Spark与Hadoop的区别1.2 Spark的应用场景1.3 Spark的作业运行流程1.4 Spark 2.X与Spark 1.X的区别1 Spark 简介与原理 Spark 是一个大规模数据处理的统一分析引擎。 具有迅速、通用、易用、支持多种资源管理器的特点。 Spark生态系统: Spark SQL…

双榜加冕!加速科技荣登2023准独角兽中国未来独角兽双榜单

4月10日至11日&#xff0c;由杭州市人民政府、民建浙江省委会、中国投资发展促进会主办的第7届万物生长大会在杭州国际博览中心隆重举行。会上&#xff0c;中国投资发展促进会创投专委会、杭州市创业投资协会联合微链共同发布2023杭州市独角兽&#xff08;准独角兽&#xff09;…

【高危】vm2 <3.9.16 沙箱逃逸漏洞(CVE-2023-29199)

漏洞描述 vm2 是一个基于 Node.js 的沙箱环境&#xff0c;可以使用列入白名单的 Node 内置模块运行不受信任的代码。 vm2 3.9.16之前版本中&#xff0c;由于transformer.js中transformer函数中异常处理逻辑不够完善&#xff0c;攻击者可通过制造异常绕过handleException()并造…

精通 TensorFlow 2.x 计算机视觉:第一部分

原文&#xff1a;Mastering Computer Vision with TensorFlow 2.x 协议&#xff1a;CC BY-NC-SA 4.0 译者&#xff1a;飞龙 本文来自【ApacheCN 深度学习 译文集】&#xff0c;采用译后编辑&#xff08;MTPE&#xff09;流程来尽可能提升效率。 不要担心自己的形象&#xff0c;…

Downie 4 4.6.14 MAC上最新最好用的一款视频下载工具

Downie for Mac 简介 Downie是Mac下一个简单的下载管理器&#xff0c;可以让您快速将不同的视频网站上的视频下载并保存到电脑磁盘里然后使用您的默认媒体播放器观看它们。 Downie 4 下载 Downie 4 for Mac Downie 4 for Mac软件特点 支持许多站点 -当前支持1000多个不同的…

printf里的格式控制符

%p&#xff1a;打印地址(指针地址)&#xff0c;十六进制形式输出&#xff0c;有多少位输出多少位&#xff0c;取决于是32 or 64位系统&#xff0c;如果输出不够位宽&#xff0c;左边补0。 &#xff05;f用来输出实数&#xff0c;以小数形式输出&#xff0c;默认情况下保留小数点…

自动化测试工程师需要具备什么技能?

如果是初入门的学习者&#xff0c;不建议拿一本书从头学&#xff0c;很可能会被里边一些专业术语和不常用的技术带偏&#xff0c;不论在公司还是在其他岗位上自学测试&#xff0c;都可以用自己搭建好的项目来练手&#xff08;如果在公司有现成的项目更好&#xff09;&#xff0…

提取图像特征方法总结 是那种很传统的方法~

目录 写在前面 一、SIFT&#xff08;尺度不变特征变换&#xff09; 1.SIFT特征提取的实质 2.SIFT特征提取的方法 3.SIFT特征提取的优点 4.SIFT特征提取的缺点 5.SIFT特征提取可以解决的问题&#xff1a; 二、HOG&#xff08;方向梯度直方图&#xff09; 1.HOG特征提取…

30岁软件测试,目前已失业4个月,迷茫不知该怎么办?

本人14年一本毕业&#xff0c;但是人特别懒&#xff0c;不爱学习&#xff0c;专业不好&#xff0c;毕业前都没找到合适工作&#xff0c;直接去创业了&#xff0c;奶茶店&#xff0c;托管&#xff0c;都弄过&#xff0c;也干过销售&#xff0c;反正浑浑噩噩度过了两年&#xff0…

CTFHub | 文件头检查

0x00 前言 CTFHub 专注网络安全、信息安全、白帽子技术的在线学习&#xff0c;实训平台。提供优质的赛事及学习服务&#xff0c;拥有完善的题目环境及配套 writeup &#xff0c;降低 CTF 学习入门门槛&#xff0c;快速帮助选手成长&#xff0c;跟随主流比赛潮流。 0x01 题目描述…

关于运行时内存数据区的一些扩展概念

栈顶缓存技术&#xff08;Top-of-Stack Cashing&#xff09; 前面提过&#xff0c;基于栈式架构的虚拟机所使用的零地址指令更加紧凑&#xff0c;但完成一项操作的时候必然需要使用更多的入栈和出栈指令&#xff0c;这同时也就意味着将需要更多的指令分派(instruction dispatc…

跨平台科学应用程序:QtiPlot 1.X Crack

QtiPlot 是一个用于数据分析和可视化的跨平台科学应用程序。由于其多语言支持&#xff0c;QtiPlot 被积极用于世界各地学术机构的教学。许多研究科学家信任 QtiPlot 来分析他们的数据并发布他们的工作结果。来自各个科学领域和行业的数千名注册用户已经选择了 QtiPlot 来帮助他…

Part-aware attention correctness for video salient object detection笔记总结

一、摘要 问题&#xff1a;在以往的VSOD中&#xff0c;一般主要是研究时空结构&#xff0c;利用隐式注意力模型去聚合相邻视频帧的互补信息。但很少有方法去关注跨视频帧的亲和力&#xff0c;即建立显式注意力图式去完成VSOD。 解决&#xff1a;提出一个新的注意力正确性策略去…

博客系统测试报告【可上线】

目录 1、测试概述 1.1、项目名称 1.2、测试时间 1.3、编写目的 1.4、测试范围 2、测试计划 2.1、测试用例 2.1.1、注册/登录模块 2.1.2、个人中心模块 2.1.3、找回密码模块 2.1.4、博客主列表模块 2.1.5、个人博客列表模块 2.1.6、个人草稿列表模块 2.1.7、博客详…