数据结构入门(C语言版)线性表带头双向循环链表接口实现

这篇具有很好参考价值的文章主要介绍了数据结构入门(C语言版)线性表带头双向循环链表接口实现。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

数据结构入门(C语言版)线性表带头双向循环链表接口实现

导航

1、带头双向循环链表介绍

数据结构入门(C语言版)线性表带头双向循环链表接口实现
在上一篇博客我们讲述了链表的概念和结构,还实现了无头单向非循环链表接口写法,那么这一章节,我们来实现另一种常用的链表组成结构——带头双向循环链表。
如果对前面的链表基本概念还是不了解,可以看作者的上一篇博客:
线性表中链表介绍及无头单向非循环链表接口实现

2、结构体及接口函数定义

首先是结构体的定义
代码如下:

typedef int LTDateType;
typedef struct ListNode
{
	LTDateType data;//结点存储元素
	struct ListNode* next;//下一结点指针
	struct ListNode* prev;//上一结点指针
}LTNode;

然后就是接口函数的定义
代码如下:

void ListInit(LTNode* phead)//哨兵位头结点初始化
LTNode* BuyListNode(LTDateType x);//动态申请结点
void ListPushBack(LTNode* phead, LTDateType x);//双向链表尾插
void ListPopBack(LTNode* phead);//双向链表尾删
void ListPushFront(LTNode* phead, LTDateType x);//双向链表头插
void ListPopFront(LTNode* phead);//双向链表头删
LTNode* ListFind(LTNode* phead, LTDateType x);//双向链表查找
void ListInsert(LTNode* pos, LTDateType x);//在pos位置前插入
void ListErase(LTNode* pos);//删除pos位置的结点
void ListPrint(LTNode* phead);//打印双向链表
void ListDestroy(LTNode* phead);//销毁双向链表

3、接口函数实现

在上一篇博客中我们讲到不带头的单非循环链表存在一定缺陷,就是无法访问上一结点,但是这一节讲的带头双向循环链表就很好的弥补了这一缺点,带头双向循环链表看来比单链表结构要复杂很多,但其实实现起来要比单链表更简单,更高效;下面我们就来实现带头双向循环链表的接口函数吧!

3.1 头结点初始化

头结点初始化(ListInit)
首先是我们的头结点的定义,它是不存放数据的,起到一个哨兵的作用
代码如下:

void ListInit(LTNode* phead)
{
	// 哨兵位头结点
	LTNode* phead = (LTNode*)malloc(sizeof(LTNode));
	phead->next = phead;
	phead->prev = phead;

	return phead;
}

第一步当然是先使用malloc函数申请内存空间,然后就是两个指针的建立,尾部指针指向头结点头部,头部指针指向头结点尾部,返回带头结点,头结点初始化完成。

3.2 结点动态内存申请

结点动态内存申请(BuyListNode)
这个函数和上一篇中的单链表的函数类似
代码如下:

LTNode* BuyListNode(LTDateType x)
{
	LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
	newnode->data = x;
	newnode->next = NULL;
	newnode->prev = NULL;
	return newnode;
}

第一步也是先使用malloc函数申请内存空间,然后就是初始化这个结点的操作,将元素插入,两个指针指向空,返回新结点,完成结点初始化操作。

3.3 双向链表尾插

双向链表尾插(ListPushBack)
代码如下:

void ListPushBack(LTNode* phead, LTDateType x)
{
	assert(phead);

	LTNode* tail = phead->prev;
	LTNode* newnode = BuyListNode(x);

	tail->next = newnode;
	newnode->prev = tail;
	newnode->next = phead;
	phead->prev = newnode;
}

这里我们先是把phead的上一级指针赋给tail
再将要插入的元素赋给临时结点newnode
接着将tail的下一级指针指向newnode
再将newnode上一级指针指向tail
newnode下一级指针指向被插入结点phead
最后将phead的上一级指针再指向newnode完成尾插操作

3.4 双向链表尾删

双向链表尾删(ListPopBack)
代码如下:

void ListPopBack(LTNode* phead)
{
	assert(phead);
	assert(phead->next != phead);//防止链表中无元素继续删除的断言

	LTNode* tail = phead->prev;
	phead->prev = tail->prev;
	tail->prev->next = phead;

	free(tail);
}

上述代码中第二个断言是为了防止链表中无元素继续删除
这里我们也是先把phead的上一级指针赋给tail
再将tail的上一级指针赋给phead的上一级指针
也就是指向要删除结点的上一结点
最后将要删除结点的前一个结点的下一级指针指向头结点
然后释放掉tail的内存空间完成尾删

3.5 双向链表头插

双向链表头插(ListPushFront)
代码如下:

void ListPushFront(LTNode* phead, LTDateType x)
{
	assert(phead);
	LTNode* newnode = BuyListNode(x);
	LTNode* next = phead->next;

	phead->next = newnode;
	newnode->prev = phead;

	newnode->next = next;
	next->prev = newnode;
}

这里我们先和尾插一样将要插入的元素值赋给临时结点newnode
将phead下一级指针赋给临时结点next
再将newnode赋给phead下一级指针
也就是把phead的尾部指针指向newnode
把phead赋给newnode的上一级指针
再将next赋给newnode的下一级指针
最后把newnode赋给next的上一级指针,完成头插

3.6 双向链表头删

双向链表头删(ListPopFront)
代码如下:

void ListPopFront(LTNode* phead)
{
	assert(phead);
	assert(phead->next != phead);//防止链表中无元素继续删除的断言

	LTNode* next = phead->next;
	LTNode* nextNext = next->next;

	phead->next = nextNext;
	nextNext->prev = phead;
	free(next);
}

和尾删一样这里的第二个断言也是为了防止链表中无元素继续删除
头删的第一步就是将phead的下一级指针赋给next
再将next的下一级指针赋给nextNext
再将nextNext赋给phead的下一级指针
最后将phead赋给nextNext的上一指针
把next的内存空间释放完成头删

3.7 双向链表查找

双向链表查找(ListFind)
代码如下:

LTNode* ListFind(LTNode* phead, LTDateType x)
{
	assert(phead);

	LTNode* cur = phead->next;
	while (cur != phead)
	{
		if (cur->data == x)
		{
			return cur;
		}
		cur = cur->next;
	}
	return NULL;
}

这里的查找就是使用一个while循环遍历链表找到某节点的data符合要查找的值
找到了便返回结点,如果遍历一遍没找到,则返回空(NULL)。

3.8 在pos位置前插入

pos位置前插入(ListInsert)
代码如下:

void ListInsert(LTNode* pos, LTDateType x)
{
	assert(pos);
	LTNode* posPrev = pos->prev;
	LTNode* newnode = BuyListNode(x);

	posPrev->next = newnode;
	newnode->prev = posPrev;
	newnode->next = pos;
	pos->prev = newnode;
}

插入函数的实现首先创建第一个临时结点posPrev
把pos的上一级指针赋给posPrev
将要插入的元素x赋给newnode
再将newnode赋给posPrev的下一级指针
再将posPrev赋给newnode的上一级指针
再将pos赋给newnode的下一级指针
最后将再将newnode赋给pos的上一级指针完成插入操作
在这里我们可以利用ListInsert函数将前面的尾插和头插进行同义替换
双向链表尾插(ListPushBack)同义替换
代码如下:

void ListPushBack(LTNode* phead, LTDateType x)
{
	assert(phead);
	ListInsert(phead, x);
}

双向链表头插(ListPushFront)同义替换
代码如下:

void ListPushFront(LTNode* phead, LTDateType x)
{
	assert(phead);
	ListInsert(phead->next, x);
}

3.9 删除pos位置的结点

删除pos位置的结点(ListErase)
代码如下:

void ListErase(LTNode* pos)
{
	assert(pos);

	LTNode* posPrev = pos->prev;
	LTNode* posNext = pos->next;

	posPrev->next = posNext;
	posNext->prev = posPrev;
	free(pos);
	pos = NULL;
}

首先将pos的上一级指针赋给posPrev
再将将pos的下一级指针赋给posNext
再将posNext赋给posPrev下一级指针
最后把posPrev赋给posNext上一级指针
将pos内存空间释放,使pos等于空(NULL),完成删除。
同样的,我们也可以利用这个ListErase函数对尾删和头删进行同义替换
双向链表尾删(ListPopBack)同义替换
代码如下:

void ListPopBack(LTNode* phead)
{
	assert(phead);
	assert(phead->next != phead);
	ListErase(phead->prev);
}

双向链表头删(ListPopFront)同义替换
代码如下:

void ListPopFront(LTNode* phead)
{
	assert(phead);
	assert(phead->next != phead);
	ListErase(phead->next);
}

3.10 打印双向链表

打印双向链表(ListPrint)
代码如下:

void ListPrint(LTNode* phead)
{
	assert(phead);

	LTNode* cur = phead->next;
	while (cur != phead)
	{
		printf("%d ", cur->data);
		cur = cur->next;
	}
	printf("\n");
}

这里的打印操作同样是利用while循环进行一遍遍历打印输出

3.11 销毁双向链表

销毁双向链表(ListDestroy)
代码如下:

void ListDestroy(LTNode* phead)
{
	assert(phead);
	LTNode* cur = phead->next;
	while (cur != phead)
	{
		LTNode* next = cur->next;
		free(cur);
		cur = next;
	}

	free(phead);
	phead = NULL;
}

和打印函数原理一样,只不过这里是再进行遍历的同时进行逐个删除
最后将phead内存空间释放,令phead等于空(NULL)完成链表销毁操作。
在这里最后讲一下断言,在之前的单链表那一节的接口函数都有,写断言是为了让代码更健壮
一旦出现了编译错误,我们可以立马排查出问题出在哪里,这是一个不错的代码习惯
带头双向循环链表接口代码可能不是那么好理解,但是实现起来时,却更方便,所以带头双向循环链表对于我们来说是非常必要的知识点!

4、顺序表和链表的区别

特征 顺序表 链表
存储空间 物理上一定连续 逻辑上连续物理上不一定连续
随机访问 支持:O(1) 不支持:O(N)
任意位置插入或删除元素 可能需要搬移元素,效率低O(N) 只需修改指针指向
插入 动态顺序表,空间不够时需要扩容 没有容量的概念
应用场景 元素高效存储+频繁访问 任意位置插入和删除频繁
缓存利用率

5、结语

顺序表到这一篇就结束了,这里的带头双向循环链表可能在代码体现上不是那么容易理解,这需要我们不断的去进行学习和实操,如果知识光看,在数据结构这门课的学习上是不会有提高的,最重要的还是练习!!!

制作不易,如有不正之处敬请指出,感谢大家的来访,UU们的观看是我坚持下去的动力,在时间的催化剂下,让我们彼此都成为更优秀的人吧!!!不要忘了一键三连呦!文章来源地址https://www.toymoban.com/news/detail-410927.html

到了这里,关于数据结构入门(C语言版)线性表带头双向循环链表接口实现的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • 【数据结构】带头双向循环链表(小白入门必备知识)

    【数据结构】带头双向循环链表(小白入门必备知识)

    前言: 💥🎈个人主页:​​​​​​Dream_Chaser~ 🎈💥 ✨✨专栏:http://t.csdn.cn/oXkBa ⛳⛳本篇内容:c语言数据结构--带头双向循环链表 目录 一.带头双向循环链表  A.带头双向循环链表概念 B.带头双向循环链表的实现 1.带头双向循环链表的结构 2.动态申请节点函数 3.链表的初始化

    2024年02月12日
    浏览(11)
  • 【数据结构】C语言实现双向链表(带头结点、循环)

    【数据结构】C语言实现双向链表(带头结点、循环)

    结点定义: 接口定义: 我们将申请结点的代码封装成函数,方便后续使用 由于是带头结点的双向链表,因此在使用链表前,我们需要对链表进行初始化。 遍历链表,值得说的是,带头结点的双向链表的循环结束条件是 cur != phead 尾插时,需要先找到尾结点,然后将新结点插

    2024年02月03日
    浏览(38)
  • 【数据结构 -- C语言】 双向带头循环链表的实现

    【数据结构 -- C语言】 双向带头循环链表的实现

    目录 1、双向带头循环链表的介绍 2、双向带头循环链表的接口 3、接口实现 3.1 开辟结点 3.2 创建返回链表的头结点 3.3 判断链表是否为空 3.4 打印 3.5 双向链表查找 3.6 双向链表在pos的前面进行插入 3.6.1 头插 3.6.2 尾插 3.6.3 更新头插、尾插写法 3.7 双向链表删除pos位置的节点

    2024年02月09日
    浏览(36)
  • 追梦之旅【数据结构篇】——详解C语言动态实现带头结点的双向循环链表结构

    追梦之旅【数据结构篇】——详解C语言动态实现带头结点的双向循环链表结构

        😎博客昵称:博客小梦 😊最喜欢的座右铭:全神贯注的上吧!!! 😊作者简介:一名热爱C/C++,算法等技术、喜爱运动、热爱K歌、敢于追梦的小博主! 😘博主小留言:哈喽! 😄各位CSDN的uu们,我是你的博客好友小梦,希望我的文章可以给您带来一定的帮助,话不

    2024年01月17日
    浏览(50)
  • 链接未来:深入理解链表数据结构(二.c语言实现带头双向循环链表)

    链接未来:深入理解链表数据结构(二.c语言实现带头双向循环链表)

    上篇文章简述讲解了链表的基本概念并且实现了无头单向不循环链表:链接未来:深入理解链表数据结构(一.c语言实现无头单向非循环链表)-CSDN博客 那今天接着给大家带来带头双向循环链表的实现 : 头文件DoubleList.h:用来基础准备(常量定义,typedef),链表表的基本框架

    2024年01月16日
    浏览(29)
  • 【数据结构】带头双向循环链表---C语言版(单链表我们分手吧,不要再找我玩了!!!)

    【数据结构】带头双向循环链表---C语言版(单链表我们分手吧,不要再找我玩了!!!)

    在前面我们学习实现了单链表(无头单向不循环链表),这里我们引入带头双向循环链表 很明显这两种链表的结构截然不同,但都是作为链表最常使用链表结构 前者因其结构上的缺点而作为面试考题的常驻嘉宾,而且复杂麻烦 后者则是以结构最优著称,实现起来也是非常的

    2024年02月10日
    浏览(37)
  • 数据结构---带头双向循环链表

    数据结构---带头双向循环链表

    什么是双向带头循环链表? 上面简单的一个非空 带头循环双向链表逻辑图 如何定义一个双向链表? 根据图和代码可以看双向链表就是单链表的每个结点中,在设置一个指向前驱节点的指针 简单认识之后,对他进行初始化(申请一个头节点,让前驱和后驱指针都指向自己) 代码

    2024年02月07日
    浏览(7)
  • 数据结构——带头双向循环链表

    数据结构——带头双向循环链表

    在创建带头双向循环链表的节点中比之前单链表节点的创建多了一个struct ListNode* prev;结构体指针,目的在与存储前一个节点的地址,便于将整个链表连在一起。 动态申请内存结点,函数返回的是一个指针类型,用malloc开辟一个LTNode大小的空间,并用node指向这个空间,再判断

    2024年02月09日
    浏览(6)
  • 带头双向循环链表--数据结构

    带头双向循环链表--数据结构

    😶‍🌫️😶‍🌫️😶‍🌫️😶‍🌫️Take your time ! 😶‍🌫️😶‍🌫️😶‍🌫️😶‍🌫️ 💥个人主页:🔥🔥🔥大魔王🔥🔥🔥 💥所属专栏:🔥魔王的修炼之路–数据结构🔥 如果你觉得这篇文章对你有帮助,请在文章结尾处留下你的 点赞 👍和 关注 💖,支持一

    2024年02月01日
    浏览(17)
  • 数据结构-带头双向循环链表

    数据结构-带头双向循环链表

    前言: 链表有很多种,上一章结,我复盘了单链表,这一章节,主要针对双链表的知识点进行,整理复盘,如果将链表分类的话,有很多种,我就学习的方向考察的重点,主要针对这两种链表进行整理。 带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用

    2023年04月09日
    浏览(14)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包