循环链表(单循环、双循环)(数据结构与算法)

循环链表:循环单链表、循环双链表

1. 循环单链表

在这里插入图片描述

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

在这里插入图片描述

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

循环单链表可以具有以下特点和优势:

  1. 尾节点的指针指向头节点,使得在遍历链表时不需要特别处理尾节点,方便实现循环遍历。
  2. 可以更容易地进行环形操作,如判断链表是否形成环、寻找环的起始点等。
  3. 循环单链表的插入和删除操作相对简单,因为不需要特别处理头部和尾部情况。
  4. 在使用循环单链表时,我们需要额外关注以下几点:
  5. 在插入和删除节点时,要确保更新指针的正确性,以避免死循环或链表中断。
  6. 在循环单链表中遍历时要设置终止条件,防止进入无限循环。

1.1 初始化一个循环单链表

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;
}

2.2 判断单链表是否为空

//判断循环单链表是否为空, 检查头结点指针是否指向它自己就行,如下图所示
bool Empty(LinkList L)
{if(L->next == L)  //检查头结点指针是否指向它自己return true;else return false;
}

在这里插入图片描述


2.3 判断 p 结点是否为循环单链表的表尾结点

//判断 p 结点是否为循环单链表的表尾结点
bool isTail(LinkList L, LNode *p)
{if(p->next == L)return true;elsereturn false;
}

在这里插入图片描述

从头结点找到尾部,时间复杂度为O(n), 从尾部找到头部,时间复杂度为O(1), 可以让L指向表尾元素(插入、删除时可能需要修改L)


2. 循环双链表

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

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

双链表:表头结点的prior指向NULL; 表尾结点的next指向NULL。

在这里插入图片描述
循环双链表:表头结点的prior指向表尾结点; 表尾结点的next指向头结点。
在这里插入图片描述

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

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

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

2.2 循环双链表的初始化

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

在这里插入图片描述

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

2.3 判断循环链表是否为空

typedef struct DNode     //创建循环双链表
{ElemType data;struct DNode *prior,*next;
}DNode, *DLinkList;void testDLinkList()
{//初始化循环双链表DLinkList L;InitDLinkList(L);//.......后续代码.......
}
//判断循环链表是否为空
bool Empty(DLinkList L)
{if(L->next == L)return true;elsereturn false;
}

2.4 判断结点p是否为循环双链表的表尾结点

bool isTail(DLinkList L, DNode  *p)
{if(p->next == L)return true;elsereturn false;
}

2.5 双链表的插入

//在p结点之后插入s结点
bool InsertNextDNode(DNode *p, DNode *s)
{s->next = p->next;	//将结点*s插入到结点*p之后p->next->prior = s;   //如果s->prior = p;p->next = s;
}

在这里插入图片描述

2.6 双链表的删除

//删除p的后继结点q
p->next = q->next;
q->next->prior = p;  //如果p没有后继结点,普通循环链表会出问题
free(q);

在这里插入图片描述

3. 知识回顾

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/160080.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

我用 ChatGPT 的探索与实践

本文主要介绍在实际的开发过程当中,如何使用GPT帮助开发,优化流程;本文所有代码和脚本均是利用GPT生成。 技术交流群 建了技术答疑、交流群!想要进交流群、资料的同学,可以直接加微信号:mlc2060。加的时候…

Leetcode-509 斐波那契数列

使用循环 class Solution {public int fib(int n) {if(n 0){return 0;}if(n 1){return 1;}int res 0;int pre1 1;int pre2 0;for(int i 2; i < n; i){res pre1 pre2;pre2 pre1;pre1 res;}return res;} }使用HashMap class Solution {private Map<Integer,Int…

项目实战:展示第一页数据

1、在FruitDao接口中添加查询第一页数据和查询总记录条数 package com.csdn.fruit.dao; import com.csdn.fruit.pojo.Fruit; import java.util.List; //dao &#xff1a;Data Access Object 数据访问对象 //接口设计 public interface FruitDao {void addFruit(Fruit fruit);vo…

软件开发项目文档系列之十如何撰写测试用例

目录 1 概述1.1 编写目的1.2 定义1.3 使用范围1.4 参考资料1.5 术语定义 2 测试用例2.1 功能测试2.1.1 用户登录功能2.1.2 商品搜索功能 2.2 性能测试2.2.1 网站响应时间2.2.2 并发用户测试 附件&#xff1a; 测试用例撰写的要素和注意事项附件1 测试用例要素附件2 测试用例的注…

大厂的 Git 代码管理规范是怎样的?

分支命名 master 分支 master 为主分支&#xff0c;也是用于部署生产环境的分支&#xff0c;需要确保 master 分支稳定性。master 分支一般由 release 以及 hotfix 分支合并&#xff0c;任何时间都不能直接修改代码。 develop 分支 develop 为开发环境分支&#xff0c;始终…

LabVIEW实现变风量VAV终端干预PID控制

LabVIEW实现变风量VAV终端干预PID控制 变风量&#xff08;VAV&#xff09;控制方法的研究一直是VAV空调研究的重点。单端PID控制在温差较大时&#xff0c;系统容易出现过冲。针对空调终端单端PID控制的不足&#xff0c;设计一种干预控制与PID控制耦合的控制方法。项目使用LabV…

Adobe:受益于人工智能,必被人工智能反噬

来源&#xff1a;猛兽财经 作者&#xff1a;猛兽财经 总结&#xff1a; &#xff08;1&#xff09;Adobe(ADBE)受益于生成式人工智能的兴起&#xff0c;其一直能实现两位数的收入增长就证明了这一点。 &#xff08;2&#xff09;在生成式人工智能兴起时&#xff0c;该公司就快…

数据结构——顺序表(SeqList)

目录 1. 顺序表介绍 2. 顺序表工程 2.1 顺序表定义 2.1.1 静态顺序表 2.1.2 动态顺序表 2.2顺序表接口 2.2.1 顺序表初始化 2.2.2 顺序表打印 2.2.3 顺序表销毁 2.2.4 顺序表数据插入 2.2.4.1 容量检查 2.2.4.2 顺序表尾插 2.2.4.3 顺序表头插 2.2.4.4 顺序表随机…

selenium自动化测试入门 —— 设置等待时间

time.sleep(3) 固定等待3秒 driver.implicitly_wait(10) 隐性的等待&#xff0c;对应全局 WebDriverWait( driver, timeout).until(‘有返回值的__call__()方法或函数’) 显性的等待&#xff0c;对应到元素 一、time.sleep(seconds) 固定等待 import time time.sleep(3) #…

UG NX机械设计软件常见安装问题

UG软件版本这里咱们就不提了&#xff0c;大部分伙伴应该都是钩子激活软件&#xff0c;肯定会遇到或多或少的安装问题&#xff0c;今天这里给大家总结了下&#xff0c;需要的小伙伴自取。 有其他问题可以一起讨论&#xff0c;也希望看到的小伙伴多关注支持哦。 安装UGNX的必要…

XSAN数据恢复-存储空间架构迁移时误格式化存储系统的XSAN数据恢复案例

XSAN数据恢复环境&#xff1a; 昆腾存储&#xff0c;MAC OS操作系统&#xff0c;存放视频类数据&#xff08;MXF、MOV等格式文件&#xff09;。 XSAN故障&检测&#xff1a; 将存储空间从XSAN架构迁移到STORNEXT架构后&#xff0c;存储空间中数据全部丢失。 故障存储中一共…

【微服务】mysql + elasticsearch数据双写设计与实现

目录 一、前言 二、为什么使用mysqles双写 2.1 单用mysql的问题 2.2 为什么不直接使用es 2.2.1 非关系型表达 2.2.2 不支持事务 2.2.3 多字段将造成性能低下 三、mysqles双写方案设计要点 3.1 全新设计 VS 中途调整架构 3.2 全表映射 VS 关键字段存储 3.2.1 最大程度…