C语言详解如何实现带头双向循环链表

目录
  • 创建链表存储结构
  • 创建结点
  • 链表的初始化
  • 双向链表的打印
  • 双向链表尾插
  • 双向链表尾删
  • 双向链表头插
  • 双向链表头删
  • 双向链表查找
  • 双向链表pos前插入结点
  • 双向链表删除pos位置的结点
  • 双向链表的销毁
  • 顺序表和链表的区别


2022042311415360.{C}{C}png" />

创建链表存储结构

我们需要创建一个结构体来存储一个链表结点的相关信息。

typedef int ListDataType;//将ListDataType先定义为int类型,根据需要可以改为不同的类型
//创建一个链表的结构体
typedef struct ListNode
{
	ListDataType data;//存储数据
	struct ListNode* next;//存储下一个结点的地址的指针
	struct ListNode* prev;//存储上一个结点的地址的指针
}ListNode;

创建结点

我们每次增加数据都要创建一个新的结点,但每次都创建就会非常的麻烦。所以我们考虑把创建一个结点这个功能封装成一个函数,每次要使用只要调用一下就可以了。

ListNode* BuyListNode(ListDataType x)
{
	ListNode* newnode = (ListNode*)malloc(sizeof(ListNode));//向内存申请一个新的结点
	if (newnode == NULL)
	{
		printf("BuyListNode::%s\n", strerror(errno));//打印结点空间申请失败的原因
		exit(-1);//终止程序
	}
	newnode->data = x;//将x放入申请的结点数据区
	newnode->next = NULL;//让新结点的next指向空
	newnode->prev = NULL;//让新结点的prev指向空
	return newnode;//返回新结点的地址
}

链表的初始化

带头双向循环链表我们对其初始化就是申请一个带哨兵位的头结点。接下来我们看初始化的两种方法。

void ListInit(ListNode** pphead)//这里我们需要对头结点进行改变,所以传二级指针
{
	assert(pphead);//检测传过来的pphead的地址是否为空
	*pphead = BuyListNode(0);//创建一个新的哨兵位头结点
	(*pphead)->next = *pphead;//让这个结点指向自己
	(*pphead)->prev = *pphead;
}
ListNode* ListInit()
{
	ListNode* phead = BuyListNode(0);//创建一个新的哨兵位头结点
	phead->next = phead;//让这个结点的next指向自己
	phead->prev = phead;//让prev指向自己
	return phead;//返回哨兵位头结点的地址
}

双向链表的打印

我们才用遍历链表的方式来打印链表中的数据。

void ListPrint(ListNode* phead)
{
	assert(phead);
	ListNode* cur = phead->next;//让cur指向哨兵位的下一个结点
	while (cur != phead)//链表打印是否结束的条件判断
	{
		printf("%d ", cur->data);
		cur = cur->next;
	}
	printf("\n");
}

双向链表尾插

双向循环链表结构比较优,所以很方便找到尾结点。尾结点就是哨兵位结点prev指向的结点。

void ListPushBack(ListNode* phead, ListDataType x)
{
	assert(phead);//检查传过来的头结点是否为空
	ListNode* tail = phead->prev;//找到链表的尾结点
	ListNode* newnode = BuyListNode(x);//创建一个新的结点

	tail->next = newnode;//让尾结点的next指向新的结点
	newnode->prev = tail;//让新结点的prev指向之前的尾结点
	newnode->next = phead;//让新的结点的next指向头结点
	phead->prev = newnode;//让头结点指向新的尾结点
}

双向链表尾删

void ListPopBack(ListNode* phead)
{
	assert(phead);
	if (phead->next == phead)//如果链表只有哨兵位结点的情况,就不能继续删除了
		return;

	ListNode* tail = phead->prev;//找到尾结点
	ListNode* tailPrev = tail->prev;//定义尾结点的前一个结点
	free(tail);//释放要删除的结点
	tailPrev->next = phead;//让前一个结点指向哨兵位结点
	phead->prev = tailPrev;//让哨兵位结点指向新的尾结点
}

双向链表头插

双向链表的头插就是在哨兵位结点的下一个位置插入一个新的数据。

注意:这一种方法种的指向关系不能随意颠倒,否则就会出错。

void ListPushFront(ListNode* phead, ListDataType x)
{
	assert(phead);
	ListNode* newnode = BuyListNode(x);//创建一个新结点
	newnode->next = phead->next;//新结点的next指向头结点的下一个结点
	phead->next->prev = newnode;//头结点的下一个结点指向新结点
	phead->next = newnode;//头结点指向新结点
	newnode->prev = phead;//新结点prev指向头结点
}

我们可以对上一种情况进行优化,定义一个next记录头结点的下一个结点的地址。这样我们就可以不用在意指向顺序的问题,可以减少出错的概率。

void ListPushFront(ListNode* phead, ListDataType x)
{
	assert(phead);
	ListNode* newnode = BuyListNode(x);//创建一个新结点
	ListNode* next = phead->next;//定义next记录头节点的下一个结点的位置
	phead->next = newnode;//头结点next指向新结点
	newnode->prev = phead;//新结点的prev指向头结点
	next->prev = newnode;//头结点的下一个结点的prev指向新阶段
	newnode->next = next;//新结点的next指向原头结点的下一个结点
}

双向链表头删

我们先找到头结点的下一个结点,然后在把它释放掉。这里需要注意的是如果链表为空,只有哨兵位头结点的情况,我们需要对其进行特殊的处理。

void ListPopFront(ListNode* phead)
{
	assert(phead);
	if (phead->next == NULL)//只有哨兵位头结点的情况
		return;
	ListNode* next = phead->next->next;//定义next指向要删除结点的下一个结点
	free(phead->next);//释放要删除的结点
	phead->next = next;//让哨兵位头结点指向要删除的下一个结点
	next->prev = phead;//让要删除的下一个结点的prev指向toujied
}

双向链表查找

若我们想要对指定的位置的结点进行相应的改变就得先找到对应的结点,所以我们将查找指定数据封装成一个函数。方便后续调用。

ListNode* ListFind(ListNode* phead, ListDataType x)
{
	assert(phead);
	ListNode* cur = phead->next;//让cur指向哨兵位头结点的下一个结点
	while (cur != phead)//链表循环是否结束的判断条件
	{
		if (cur->data == x)
			return cur;
		cur = cur->next;
	}
	return NULL;//找不到对于的结点
}

双向链表pos前插入结点

推荐使用第二种方法,不容易出错。

void ListInsert(ListNode* pos, ListDataType x)
{
	assert(pos);
	ListNode* newnode = BuyListNode(x);//创建一个新的结点
	pos->prev->next = newnode;//pos的前一个结点的next指向新结点
	newnode->prev = pos->prev;//新结点的prev指向pos的前一个结点
	newnode->next = pos;//新结点的next指向pos结点
	pos->prev = newnode;//pos的prev指向新结点
}
void ListInsert(ListNode* pos, ListDataType x)
{
	assert(pos);
	ListNode* newnode = BuyListNode(x);//创建一个新的结点
	ListNode* posPrev = pos->prev;//定义pos的前一个结点
	posPrev->next = newnode;//让pos的pos前一个结点的next指向新结点
	newnode->prev = posPrev;//新结点的prev指向pos的前一个结点
	newnode->next = pos;//新结点的next指向pos
	pos->prev = newnode;//pos的prev指向新结点
}

双向链表删除pos位置的结点

void ListErase(ListNode* pos)
{
	assert(pos);
	ListNode* prev = pos->prev;//prev为pos的前一个结点
	ListNode* next = pos->next;//next为pos的后一个结点
	free(pos);//释放posjied
	prev->next = next;
	next->prev = prev;
}

双向链表的销毁

void ListDestroy(ListNode* phead)
{
	assert(phead);
	ListNode* cur = phead->next;//指向哨兵位结点的下一个结点
	while (cur != phead)//依次循环找链表的每一个结点
	{
		ListNode* next = cur->next;//记录cur的下一个结点位置
		free(cur);//释放cur位置的结点
		cur = next;//指向下一个结点
	}
	free(phead);//释放哨兵位头结点
}

注意:在写双向链表的头插,头删,尾插,尾删时我们可以复用双向链表的任意位置插入和任意位置删除那两个函数。那两个函数就可以完成头插,头删,尾插,尾删这几个功能。

顺序表和链表的区别

顺序表优点:

1.物理空间是连续的,方便用下标随机访问。

2.CPU高速缓存命中率会更高。

顺序表缺点:

1.由于需要物理空间连续,空间不够需要扩容。扩容本身有一定消耗。其次扩容机制还存在一定的空间浪费。

2.头部或者中部插入删除,挪动数据,效率低。O(N)

链表优点:

1.任意位置插入删除数据效率高。O(1)

2.按需申请和释放空间

链表缺点:

不支持下标的随机访问。有些算法不适合在它上面运行。如:二分查找、排序等。

关于CPU相关的知识可以参考这一篇文章:CPU缓存知识

到此这篇关于C语言详解如何实现带头双向循环链表的文章就介绍到这了,更多相关C语言带头双向循环链表内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C语言实现带头双向循环链表

    目录 前言 1. 创建结构体 2.malloc新节点 3.创建哨兵位节点 4.尾插 5.打印 6.尾删 7.头插 8.在指定位置pos的前面进行插入 9. 删除指定位置pos节点 10.销毁链表 前言 在实际生活中最常用的就是这两种链表.无头单向非循环链表.和带头双向循环链表.无头单向非循环链表:结构简单,一般不会单独用来存数据.实际中更多是作为其他数据结构的子结构,如哈希桶.图的邻接表等等.另外这种结构在笔试面试中出现很多.带头双向循环链表:结构最复杂,一般用在单独存储数据.实际中使用的链表数

  • C语言实现带头双向循环链表的接口

    本文实例为大家分享了C语言实现带头双向循环链表的接口,供大家参考,具体内容如下 各函数功能如下 申请空间 ListNode* BuyListNode(LTDataType x) { ListNode* node = (ListNode*)malloc(sizeof(ListNode)); node->next = NULL; node->prev = NULL; node->data = x; return node; } 初始化 ListNode* ListInit() { ListN

  • C语言超详细讲解数据结构中双向带头循环链表

    目录 一.概念 二.必备工作 2.1.创建双向链表结构 2.2.初始化链表 2.3.动态申请节点 2.4.打印链表 2.5.销毁链表 三.主要功能 3.1.在pos节点前插入数据 尾插 头插 3.2.删除pos处节点数据 尾删 头删 3.3.查找数据 四.总代码 List.h 文件 List.c 文件 Test.c 文件 五.拓展 一.概念 前文我们已经学习了单向链表,并通过oj题目深入了解了带头节点的链表以及带环链表,来画张图总体回顾下: 在我们学习的链表中,其实总共有8种,都是单双向和带不带

  • C语言编程数据结构带头双向循环链表全面详解

    目录 前言 一.什么是带头循环双向链表 二.链表初始化 三.链表接口函数 1.尾插 2.头插 3.头删 4.尾删 5.任意位置插入数据 6.任意位置删除数据 四.打印链表 总结 前言 上一篇数据结构专栏:C语言数据结构单链表接口函数全面讲解教程 我们介绍了单链表的各个接口函数,大家可能会发现单链表存在一些缺陷:比如它一个节点要存储数据+下一个节点地址,占用的空间要远多于顺序表:并且由于单链表是无法从后往前找的,如果你想进行尾删这样的操作,你必须从第一个节点往后找,你的时间复杂度一定是O(n).

  • C语言手把手带你掌握带头双向循环链表

    目录 前言 带头双向循环链表的结构 代码操作 前言 关于链表这一块,写了多篇博客,学习了顺序表.单链表.及其一些练习题 顺序表:传送门:顺序表 单链表:传送门:单链表1   链表2 链表OJ:传送门:链表OJ 今天,我又来水一水博客, 介绍关于双链表. 带头双向循环链表的结构 实际上,单链表也存在一个比较大的缺陷: 1.不能从后往前遍历 2.无法找到前驱 除了单链表之外,我们自然还有双向链表,我们要说的就是带头双向循环链表,简单理解为:带头结点的,有两个方向的.循环的.结构图如下: 结构虽然比较

  • C语言 超详细介绍与实现线性表中的带头双向循环链表

    目录 一.本章重点 二.带头双向循环链表介绍 2.1什么是带头双向循环链表? 2.2最常用的两种链表结构 三.带头双向循环链表常用接口实现  3.1结构体创建 3.2带头双向循环链表的初始化  3.3创建新节点 3.4尾插 3.5打印链表 3.6头插 3.7尾删 3.8头删 3.9查找data(返回data的节点地址) 3.10在pos位置之前插入节点 3.11删除pos位置的节点 四.实现接口总结 五.在线oj训练与详解 一.本章重点 带头双向循环链表介绍 带头双向循环链表常用接口实现 实现接

  • C语言详解如何实现带头双向循环链表

    目录 创建链表存储结构 创建结点 链表的初始化 双向链表的打印 双向链表尾插 双向链表尾删 双向链表头插 双向链表头删 双向链表查找 双向链表pos前插入结点 双向链表删除pos位置的结点 双向链表的销毁 顺序表和链表的区别 2022042311415360.{C}{C}png" /> 创建链表存储结构 我们需要创建一个结构体来存储一个链表结点的相关信息. typedef int ListDataType;//将ListDataType先定义为int类型,根据需要可以改为不同的类型 //创

  • C语言中带头双向循环链表基本操作的实现详解

    目录 一.概念与结构 二.基本操作的实现 1.创建结点 2.初始化链表 3.打印链表 4.尾插 5.尾删 6.头插 7.头删 8.查找某个数并返回其指针 9.在某个位置之前插入 10.删除某个位置 11.判断链表是否为空 12.计算链表中有效值的个数 13.销毁链表 三.测试代码 一.概念与结构 无头单向非循环链表结构简单,一般不会单独用来存数据.实际中更多的是作为其他数据结构的子结构,如哈希桶.图的邻接表等等.而带头双向循环链表的结构较为复杂,一般用在单独存储数据.实际中使用的链表数据结构,都

  • C++实现带头双向循环链表的示例详解

    目录 一.双向循环链表与顺序表的区别 二.List.h 三.List.c 1.带头双向循环链表的初始化 2.带头双向循环链表的销毁 3.带头双向循环链表的打印 4.动态开辟一个节点 5.带头双向循环链表的判空 6.带头双向循环链表的尾插.尾删 7.带头双向循环链表的头插.头删 8.带头双向循环链表的长度 9.带头双向循环链表的查找.任意位置插入.删除 一.双向循环链表与顺序表的区别 不同点 顺序表 双向带头循环链表 在内存中的存储方式 连续存储 不一定连续 随机访问 支持随机访问 不支持随访问

  • C语言详解无头单向非循环链表各种操作方法

    目录 链表引入 链表介绍 创建链表 打印链表 创建结点 单链表尾插 单链表头插 单链表尾删 单链表头删 在pos位置之前插入数据 在pos位置之后插入数据 删除pos位置结点 删除pos位置之后的结点 销毁链表 链表查找 链表引入 问:上次我们看了顺序表,那么顺序表有些什么优缺点呢? 优点: 顺序表是连续的物理空间,方便下标的随机访问. 缺点: 1.增容需要申请新空间,拷贝数据,释放旧的空间.会有一定消耗. 2.头部或者中间位置插入或者删除数据,需要挪动数据,效率较低. 3.可能存在一定空间占用

  • C语言详解如何应用模拟字符串和内存函数

    目录 1.strlen 求字符串长度 使用案例: 1.计数法 2.不创建临时变量计数器-递归 3.指针-指针的方式 2.长度不受限制的字符串函数 1.strcpy 使用案例: 模拟实现: 2.strcat 使用案例: 模拟实现: 3.strcmp-比较字符串首字母的大小 使用案例: 模拟实现: 3.长度受限制的字符串函数  1.strncpy 使用案例: 2.strncat  使用案例: 3.strncmp 使用案例: 4.strstr-找子串  使用案例: 模拟实现: 5.strtok 用法:

随机推荐