拓展单向循环链表

909 字
5 分钟
拓展单向循环链表

—————————————本文旨在讨论计算机知识欢迎指正——————————————— 书接上回:我们已经了解了链表如何编写与前置节点和头指针两种表示方式,下面,我们来了解进阶写法———如何实现单向循环链表。 下面,我们来梳理一下循环链表的实现方式: 这是朴素的链表实现形式: 这是我们理想中的循环链表实现形式: 这就是我们大体的思路,然后,我们将要实现它:

  • 首先,我们第一个难点就是———如何来构造链表的结构体;因为我们需要循环,那么我们必须要知道尾巴在哪里,所以结构体里必须有tail这一容器,然后我们还要考虑如果每次我们都用一个迭代器去迭代遍历找尾巴tail的话,效率会非常低,我们需要一个更好的表示形式
// 定义节点结构
typedef int Element;
typedef struct _loop_node {
Element val;//数据域
struct _loop_node *next;//地址域,指向下一个
} LoopNode;
// 定义单向循环链表的头结构
typedef struct {
LoopNode header;//头结点
LoopNode *tail;//尾巴的容器,相当于一个移动指针迭代器;
int num;//计数器,用来遏制循环次数,防止无限循环
} LinkLoopList;

于是,笔者想到了两种结构,分别是头插法和尾插法: 1、头插法:

int insertLinkLoopHeader(LinkLoopList* link_loop, Element value) {
// 1. 先有新节点
LoopNode *node = malloc(sizeof(LoopNode));
if (node == NULL) {
return -1;
}
node->val = value;//新节点赋值
node->next = link_loop->header.next;//插到头结点的下一个元素的前面;
link_loop->header.next = node;//更新头结点在新节点的前面
// 2. 判断尾指针是否需要更新
if (link_loop->tail== &link_loop->header) {
link_loop->tail = node;
}
++link_loop->num;
return 0;
}

2、尾插法:

int insertLinkLoopTail(LinkLoopList* link_loop, Element value) {
// 1. 先有新节点
LoopNode *node = malloc(sizeof(LoopNode));
if (node == NULL) {
return -1;
}
node->val = value;
node->next = link_loop->tail->next;//尾部插入
link_loop->tail->next = node;//与原尾部节点衔接
link_loop->tail = node;//更新尾部
++link_loop->num;//更新数量
return 0;
}

然后我们从头考虑初始化和声明结构体:

// 定义节点结构
typedef int Element;
typedef struct _loop_node {
Element val;
struct _loop_node *next;
} LoopNode;
// 定义单向循环链表的头结构
typedef struct {
LoopNode header;
LoopNode *rear;
int num;
} LinkLoopList;

这是初始化:

void initLinkLoopList(LinkLoopList* link_loop) {
link_loop->header.next = link_loop->tail = &link_loop->header;//头指针和尾指针同时指向头结点
link_loop->num = 0;
}

删除:

int deleteLinkLoopList(LinkLoopList* link_loop, Element value) {
LoopNode *node = &link_loop->header;
while (node->next != &link_loop->header && node->next->val != value) {
node = node->next;
}
if (node->next->val == value) {
LoopNode *tmp = node->next;
node->next = tmp->next;
free(tmp);//释放空间,删除操作
--link_loop->num;
} else {
printf("No %d element!\n", value);
}
return 0;
}

呈现操作:

void showLinkLoopList(const LinkLoopList* link_loop) {
LoopNode *node = link_loop->header.next;
while (node != &link_loop->header) {
printf("\t%d", node->val);
node = node->next;
}
printf("\n");
}

销毁函数:

void destroyLinkLoopList(LinkLoopList* link_loop) {
LoopNode *node = link_loop->header.next;
while (node != &link_loop->header) {
LoopNode *tmp = node;
node = node->next;
free(tmp);
--link_loop->num;
}
printf("Table %d element!\n", link_loop->num);
}

最后,是测试函数:

void test01() {
LinkLoopList table;
initLinkLoopList(&table);
for (int i = 0; i < 10; ++i) {
insertLinkLoopRear(&table, i + 100);
}
showLinkLoopList(&table);
printf("======================\n");
deleteLinkLoopList(&table, 106);
showLinkLoopList(&table);
destroyLinkLoopList(&table);
}
/* 以下是几组约瑟夫环的测试答案,包括每个被杀者的顺序编号和最后的幸存者:
* 当n=5,k=2时,被杀者的顺序编号为2, 4, 1, 5,最后的幸存者是3。
* 当n=10,k=3时,被杀者的顺序编号为3, 6, 9, 2, 7, 1, 8, 5, 10,最后的幸存者是4。
* 当n=7,k=2时,被杀者的顺序编号为2, 4, 6, 1, 5, 3,最后的幸存者是7。
* 当n=10,k=17时,最后的幸存者是3
*/
void test_Joseph() {
}
int main() {
test01();
return 0;
}

————————完结撒花!!!——————希望对你有所帮助!

支持与分享

如果这篇文章对你有帮助,欢迎分享给更多人或打赏支持!

打赏
拓展单向循环链表
https://firefly.cuteleaf.cn/posts/knowledge/拓展单向循环链表/
作者
Firefly
发布于
2026-06-21
许可协议
CC BY-NC-SA 4.0
相关文章 智能推荐
1
附录对于头结点单向链表的优化方法
数据结构 ————————————本文旨在交流学习计算机知识,欢迎交流讨——————————— 上一章,我们浅显的讨论了链表的结构,本文在上一章的链表的基础上做出了些许优化,可配合食用。 回到正题,我们已经了解链表的结构,它是一种非连续空间存储的线性表,主要依靠地址域访问下一个元素并通过遍历操作进行增删改查。...
2
链表的实现与介绍
数据结构 ——————————————本文旨在交流计算机知识,欢迎指正————————————— 首先,我们回顾一下上一期讲的顺序表: 顺序表是一种物理结构上连续的,一整块连续空间的逻辑表,但是,试想一下,如果每次我们都是申请在一块整块空间里面的顺序表,但是我们并不知道堆空间里面我们malloc一块空间之后我...
3
链表最终章双向链表及其应用
数据结构 ———————————本文旨在交流探讨计算机知识,欢迎交流指正———————————— 上一章,我们介绍了链表的循环扩展,但是,单向链表毕竟是单向查询的,就算是经过循环来查找,终究是效率偏低,下面,我们来介绍一种工程项目中运用更加广泛的一种链表结构:双向链表——: 根据此结构,我们可以发现,虽然代...
4
顺序表的实现与介绍及部分工程思想的思考
数据结构 ——————————本文旨在讨论和探究计算机知识,欢迎指正———————————— 首先,我们来聊聊什么是线性表: 线性表,顾名思义,是一种具有线性结构的数据排布形式 ,是一个类似数组的东西,在逻辑上一个接一个有序的整块链接,区别于链表的无序分布链接。 如图,我们可以得出,顺序表是一条线一样的结构...
5
数据结构之树及树的存储
数据结构 ——————————————本栏目旨在交流计算机知识————————————————— 上一章节,我们介绍了线性的数据结构:表,栈,队列。接下来,我们将进入下一章节——树结构的学习。 首先,我们先介绍一下树的基本概念: 结点:使⽤树结构存储的每一个数据元素都被称为“结点”。例如图中的A就是一个结点。...
随机文章 随机推荐
Profile Image of the Author
Firefly
Hello, I'm Firefly.
公告
欢迎来到我的博客!这是一则示例公告。
音乐
封面

音乐

暂未播放

0:00 0:00
暂无歌词
分类
标签
站点统计
文章
33
分类
7
标签
25
总字数
56,127
运行时长
0
最后活动
0 天前
站点信息
构建平台
Vercel
博客版本
Firefly v6.12.3
文章许可
CC BY-NC-SA 4.0

文章目录