循环链表(包括循环单链表、循环双链表)

一、循环单链表与循环双链表的对比

二、循环单链表

List.h

cpp 复制代码
#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>

// 实现带头结点的循环单链表

// 定义结点的结构
typedef int ElemType;
typedef struct LNode
{
	ElemType data;
	struct LNode* next;
} LNode, * LinkList;


// 循环单链表的初始化
bool InitList(LinkList& L); // L是头结点的地址的别名

// 判断循环单链表是否为空(即链表中是否存储了有效数据),若为空返回true,否则返回false
bool Empty(LinkList L); // L接收头结点的地址

// 判断p指向的结点是否为循环单链表的尾结点,如果是返回true,否则返回false
bool isTail(LinkList L, LNode* p); // L接收头结点的地址

List.cpp

cpp 复制代码
#define _CRT_SECURE_NO_WARNINGS 1
#include"List.h"
// 循环单链表的初始化
bool InitList(LinkList& L) // L是头结点的地址的别名
{
	// 开辟头结点的空间
	L = (LNode*)malloc(sizeof(LNode));
	if (L == NULL) // 表示头结点的空间开辟失败
		return false;
	L->next = L; // 为了构成一个环,头结点的next指针需要指向头结点,
	return true;
}

// 判断循环单链表是否为空(即链表中是否存储了有效数据),若为空返回true,否则返回false
bool Empty(LinkList L) // L接收头结点的地址
{
	if (L->next == L) // 当头结点的next指针指向头结点时,表示链表为空
		return true;
	else
		return false;
}

// 判断p指向的结点是否为循环单链表的尾结点,如果是返回true,否则返回false
bool isTail(LinkList L, LNode* p) // L接收头结点的地址
{
	if (p->next == L) //循环单链表中,尾结点的next指针指向头结点
		return true;
	else
		return false;
}

Test.cpp

cpp 复制代码
#define _CRT_SECURE_NO_WARNINGS 1
#include"List.h"


void test()
{
	// 创建一个头指针(即指向头结点的指针)
	LNode* p = NULL;
	// 测试循环双链表的初始化
	InitList(p);
}

int main()
{
	test();
	return 0;
}

三、循环双链表

List.h

cpp 复制代码
#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
// 实现带头结点的循环双链表

typedef int ElemType;
typedef struct DNode
{
	ElemType data; // 结点中的数据域,存储一个整数
	struct DNode* prior; // 指向前一个结点
	struct DNode* next; // 指向后一个结点
}DNode, * DLinkList;
// 将struct DNode重命名为DNode,将struct DNode*重命名为DLinkList
// 因此DNode*与DLinkList完全等价


// 循环双链表的初始化
bool InitDLinkList(DLinkList& L); // L是头结点的地址的别名

// 判断循环双链表是否为空(即链表中是否存储了有效数据)。若为空返回true,否则返回false
bool Empty(DLinkList L); // L接收头结点的地址

// 判断p指向的结点是否为尾结点。若是尾结点返回true,否则返回false
bool isTail(DLinkList L, DNode* p);

// 在p指向的结点后插入s指向的结点
bool InsertNextDNode(DNode* p, DNode* s);

// 删除p指向的结点的后面一个结点
bool DeleteNextDNode(DNode* p);

List.cpp

cpp 复制代码
#define _CRT_SECURE_NO_WARNINGS 1
#include"List.h"
// 循环双链表的初始化
bool InitDLinkList(DLinkList& L) // L是头结点的地址的别名
{
	// 开辟头结点的空间
	L = (DNode*)malloc(sizeof(DNode)); // L指向头结点
	if (L == NULL) // 表示头结点的空间开辟失败
		return false;
	L->prior = L->next = L; // 为了构成一个环状,需要让头结点的next指针与prior指针均指向头结点
	return true;
}

// 判断循环双链表是否为空(即链表中是否存储了有效数据)。若为空返回true,否则返回false
bool Empty(DLinkList L) // L接收头结点的地址
{
	if (L->next = L) // 当头结点的next指针指向头结点时,表示循环双链表为空
		return true;
	else
		return false;
}

// 判断p指向的结点是否为尾结点。若是尾结点返回true,否则返回false
bool isTail(DLinkList L, DNode* p)
{
	if (p->next == L) // 尾结点的next指针指向头结点
		return true;
	else
		return false;
}


// 在p指向的结点后插入s指向的结点
bool InsertNextDNode(DNode* p, DNode* s)
{
	if (p == NULL || s == NULL) // 当p或s指向的结点无效时,无法指向删除操作
		return false;
	s->next = p->next;
	p->next->prior = s; // 如果实现的是双链表,这句代码在尾结点后插入新结点时,会导致对空指针的解引用。
	s->prior = p;
	p->next = s;
	return true;
}

// 删除p指向的结点的后面一个结点
bool DeleteNextDNode(DNode* p) // 
{
	if (p == NULL) // 当p指向的结点无效,,不能指向删除操作
		return false;
	DNode* q = p->next; // s指向待删除的结点
	p->next = q->next;
	q->next->prior = p; // 如果实现的是双链表,这句代码在删除尾结点时,会导致对空指针的解引用。
	free(q);
	q = NULL;
	return true;
}

Test.cpp

cpp 复制代码
#define _CRT_SECURE_NO_WARNINGS 1
#include"List.h"
void test()
{
	// 创建一个头指针(即指向头结点的指针)
	DNode* p = NULL;
	// 测试循环双链表的初始化
	InitDLinkList(p);
}

int main()
{
	test();

	return 0;
}
相关推荐
代码探秘者2 小时前
【Redis】告别锁失效:RedLock 与 ZooKeeper 分布式锁原理与实战对比
java·数据结构·redis·后端·python·zookeeper·面试
8Qi82 小时前
环形链表刷题笔记(LeetCode热题100--141、142)
c语言·数据结构·c++·算法·leetcode·链表
一叶落4382 小时前
139. 单词拆分(Word Break)
c语言·数据结构·算法·leetcode·深度优先·图论
Yupureki2 小时前
《算法竞赛从入门到国奖》算法基础:数据结构-单调队列
c语言·数据结构·c++·算法
小指纹2 小时前
每日一题--Tokitsukaze and Colorful Chessboard【二分】
数据结构·c++·算法
汉克老师2 小时前
GESP 四级C++考试2025年3月第二部分判断题(1-10题)
数据结构·c++·排序算法·指针·结构体·gesp4级·gesp四级
8Qi82 小时前
LeetCode热题100--189
c语言·数据结构·c++·算法·leetcode
进击的雷神2 小时前
请求频率限制、嵌套数据结构、多目录聚合、地址字段重构——K展爬虫四大技术难关攻克纪实
数据结构·爬虫·python·重构
Flying pigs~~3 小时前
我的leetcode hot100之行(持续更新)
数据结构·算法·leetcode