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

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

二、循环单链表

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;
}
相关推荐
Titan202414 小时前
map和set的封装学习笔记
数据结构·c++
Yupureki14 小时前
《算法竞赛从入门到国奖》算法基础:动态规划-路径dp
数据结构·c++·算法·动态规划
重庆小透明16 小时前
力扣刷题【3】相交链表
算法·leetcode·链表
算法鑫探16 小时前
C语言实战:学生成绩统计与分析
c语言·数据结构·算法·新人首发
_日拱一卒17 小时前
LeetCode:最小覆盖字串
java·数据结构·算法·leetcode·职场和发展
郝学胜-神的一滴17 小时前
Qt6 + OpenGL 3.3 渲染环境搭建全指南:从空白窗口到专属渲染画布的优雅实现
数据结构·c++·线性代数·算法·系统架构·图形渲染
小肥米17 小时前
分块查找ASL公式推导,为什么是两个ASL之和
数据结构·算法
样例过了就是过了17 小时前
LeetCode热题100 最小栈
数据结构·c++·算法·leetcode
计算机安禾17 小时前
【数据结构与算法】第18篇:数组的压缩存储:对称矩阵、三角矩阵与稀疏矩阵
c语言·开发语言·数据结构·c++·线性代数·算法·矩阵
迈巴赫车主17 小时前
错位排序算法
开发语言·数据结构·算法·排序算法