【数据结构】二、线性表:4.循环链表的定义及其基本操作(循环单链表,循环双链表的初始化、判空、判断头结点、尾结点、插入、删除)

文章目录

4.循环链表

4.1循环单链表

循环单链表(Circular Singly Linked List)是一种特殊类型的单链表,其中最后一个节点的指针指向头节点,形成一个循环。

循环单链表与普通单链表的主要区别在于,循环单链表的尾节点的指针不是指向 nullptr,而是指向头节点,形成一个闭环。这意味着,在循环单链表中,可以通过尾节点的指针重新回到头节点。

循环单链表的特点和优势:

  1. 尾节点的指针指向头节点,使得在遍历链表时不需要特别处理尾节点,方便实现循环遍历。
  2. 可以更容易地进行环形操作,如判断链表是否形成环、寻找环的起始点等。
  3. 循环单链表的插入和删除操作相对简单,因为不需要特别处理头部和尾部情况。
  4. 在使用循环单链表时,我们需要额外关注以下几点:
  5. 在插入和删除节点时,要确保更新指针的正确性,以避免死循环或链表中断。
  6. 在循环单链表中遍历时要设置终止条件,防止进入无限循环。
4.1.1初始化
c 复制代码
typedef struct LNode{  		//定义单链表结合类型
	ElemType data;			//每个结点存放一个一个数据元素
	struct LNode *next; 	//指针指向下一个结点
}LNode, *LinkList;

//初始化一个循环单链表
bool InitList(LinkList &L){
	L=(LNode *)malloc(sizeof(LNode));	//分配一个头结点
	if(L == NULL)  		//内存不足,分配失败
		return false;
	L->next = L;		//头结点next指向头结点
	return true;
}
4.1.2判断单链表是否为空

判断循环单链表是否为空,检查头结点指针是否指向它自己就行

c 复制代码
bool Empty(LinkList L){
	if(L->next == L)  //检查头结点指针是否指向它自己
		return true;
	else 
		return false;
}
4.1.3判断p结点是否为循环单链表的表尾结点
c 复制代码
//判断 p 结点是否为循环单链表的表尾结点
bool isTail(LinkList L, LNode *p)
{
	if(p->next == L)
		return true;
	else
		return false;
}
4.2循环双链表

循环双链表(Circular Doubly Linked List)是一种特殊类型的双向链表,其中最后一个节点的下一个指针指向头节点,头节点的前一个指针指向最后一个节点,形成一个循环。

循环双链表与普通双链表的主要区别在于,循环双链表既具有双向链表的前驱和后继关系,也具有循环遍历的能力。

循环双链表的特点和优势:

  1. 可以通过任意节点的前驱和后继指针方便地在双链表中进行插入和删除操作。
  2. 最后一个节点的下一个指针指向头节点,使得在遍历链表时不需要特别处理尾节点,可以很方便地实现循环遍历。
  3. 可以更容易地进行环形操作,如判断链表是否形成环、寻找环的起始点等。
  4. 循环双链表的插入和删除操作相对简单,不需要特别处理头部和尾部情况。

注意:在插入和删除节点时,要确保更新前驱和后继指针的正确性,以避免链表中断或形成其他错误结构。在循环双链表中遍历时要设置终止条件,防止进入无限循环。

4.2.1初始化

当我们在初始化一个双链表时,需要让头结点的前指针和后指针都指向头结点自己(而普通的双链表指向NULL)

c 复制代码
typedef struct DNode{
	ElemType data;
	struct DNode *prior,*next;
}DNode, *DLinkList;

//初始化空的循环双链表
bool InitDLinkList(DlinkList &L){
	L = (DNode*) malloc(sizeof(DNode));	//分配一个头结点
	if(L == NULL)		//内存不足,分配失败
		return false;
	L->prior = L;		//头结点的prior指向头结点
	L->next = L; 		//头结点的next指向头结点
	return true;		//初始化成功
}
4.2.2判断循环链表是否为空

检查头结点next指针是否指向它自己

c 复制代码
//判断循环链表是否为空
bool Empty(DLinkList L){
	if(L->next == L)
		return true;
	else
		return false;
}
4.2.3判断结点p是否为循环双链表的表尾结点

检查头结点next指针是否指向头结点

c 复制代码
bool isTail(DLinkList L, DNode  *p){
	if(p->next == L)
		return true;
	else
		return false;
}
4.2.4双链表的插入
c 复制代码
//在p结点之后插入s结点
bool InsertNextDNode(DNode *p, DNode *s){
	s->next = p->next;
	p->next->prior = s;//如果p没有后继结点,普通链表会出问题
	s->prior = p;
	p->next = s;
}
4.2.5双链表的删除
c 复制代码
//删除p的后继结点q
p->next = q->next;
q->next->prior = p;  //如果p没有后继结点,普通链表会出问题
free(q);
相关推荐
类球状22 分钟前
顺序表 —— OJ题
算法
Miraitowa_cheems1 小时前
LeetCode算法日记 - Day 11: 寻找峰值、山脉数组的峰顶索引
java·算法·leetcode
指针满天飞1 小时前
Collections.synchronizedList是如何将List变为线程安全的
java·数据结构·list
CoovallyAIHub1 小时前
方案 | 动车底部零部件检测实时流水线检测算法改进
深度学习·算法·计算机视觉
CoovallyAIHub1 小时前
方案 | 光伏清洁机器人系统详细技术实施方案
深度学习·算法·计算机视觉
lxmyzzs1 小时前
【图像算法 - 14】精准识别路面墙体裂缝:基于YOLO12与OpenCV的实例分割智能检测实战(附完整代码)
人工智能·opencv·算法·计算机视觉·裂缝检测·yolo12
洋曼巴-young1 小时前
240. 搜索二维矩阵 II
数据结构·算法·矩阵
汉汉汉汉汉1 小时前
C++11新特性详解:从列表初始化到线程库
c++
丑小鸭是白天鹅2 小时前
嵌入式C语言学习笔记之枚举、联合体
c语言·笔记·学习
GUET_一路向前2 小时前
【C语言防御性编程】if条件常量在前,变量在后
c语言·开发语言·if-else·防御性编程