数据结构与算法教程,数据结构C语言版教程!(第二部分、线性表详解:数据结构线性表10分钟入门)九

 第二部分、线性表详解:数据结构线性表10分钟入门

线性表,数据结构中最简单的一种存储结构,专门用于存储逻辑关系为"一对一"的数据。

线性表,基于数据在实际物理空间中的存储状态,又可细分为顺序表(顺序存储结构)和链表(链式存储结构)。

本章还会讲解顺序表和链表的结合体——静态链表,不仅如此,还会涉及循环链表、双向链表、双向循环链表等链式存储结构。

十七、如何判断单链表为有环链表?

循环链表一节,给大家详细地介绍了循环链表。在此基础上,本节带领大家讨论一个问题:如何判断一个单链表中有环?

注意,有环链表并不一定就是循环链表。循环链表指的是“首尾相连”的单链表,而有环链表则指的是单链表中存在一个循环子链表,如图 1 所示。

图 1 有环链表示意图

图 1 所示就是一个有环链表,但并不是循环链表。

那么,如果给定一个单链表,如何判断其是否为有环链表呢?常用的判断方法有如下 2 种。

1) 最直接的实现思想就是:从给定链表的第一个节点开始遍历,每遍历至一个节点,都将其和所有的前驱节点进行比对,如果为同一个节点,则表明当前链表中有环;反之,如果遍历至链表最后一个节点,仍未找到相同的节点,则证明该链表中无环。

注意,如果一个单链表为有环链表,基于单链表中各节点有且仅有 1 个指针域的特性,则势必该链表是没有尾结点的(如图 1 所示)。换句话说,有环链表的遍历过程是无法自行结束的,需要使用 break 语句手动结束遍历。

基于上面的实现思想,下面设计了一个相应的实现函数:

//自定义 bool 类型

typedef enum bool

{

        False=0,

        True=1

}bool;

// H 为链表的表头

bool HaveRing(link * H) {

        link * Htemp = H; /

        /存储所遍历节点所有前驱节点的存储地址,64位环境下地址占 8 个字节,所以这里用         long long 类型

        long long addr[20] = { 0 };

        int length = 0, i = 0;

        //逐个遍历链表中各个节点

        while (Htemp) {

                //依次将 Htemp 和 addr 数组中记录的已遍历的地址进行比对

                for (i = 0; i < length; i++) {

                        //如果比对成功,则证明有环

                        if (Htemp == addr[i]) {

                                return True;

                        }

                }

                //比对不成功,则记录 Htemp 节点的存储地址

                addr[length] = Htemp;

                length++;

                Htemp = Htemp->next;

        }

        return False;

}

如上所示,当函数的返回值为 True,表示该链表有环;反之若函数返回值为 False,表明链表中无环。显然,此实现方案的时间复杂度为O(n^{2})

2) 相比上一种实现方案,这里介绍一种时间复杂度为 O(n) 的算法。

该算法的实现思想需要借助一个论点,即在一个链表中,如果 2 个指针(假设为 H1 和 H2)都从表头开始遍历链表,其中 H1 每次移动 2 个节点的长度(H1 = H1->next->next),而 H2 每次移动 1 个节点的长度(H2 = H2->next),如果该链表为有环链表,则 H1、H2 最终必定会相等;反之,如果该链表中无环,则 H1、H2 永远不会相遇。

有关在有环链表中 H1 和 H2 必定会相遇的结论,假设有环链表中的环包含 n 个节点,则第一次遍历,H1 和 H2 相差 1 个节点;第二次遍历,H1 和 H2 相差 2 个节点;第三次遍历,H1 和 H2 相差 3 个节点...,最终经过多次遍历,H1 和 H2 会相差 n-1 个节点,此时就会在环中重合,此时 H1 和 H2 相等。

基于以上这个结论,我们可以轻松编写出对应的实现代码:

//H为链表的表头,该函数会返回一个枚举的 bool 类型数据

bool HaveRing(link * H) {

        link * H1 = H->next;

        link * H2 = H;

        while (H1) {

                if (H1 == H2) {

                        //链表中有环

                        return True;

                } else {

                        H1 = H1->next;

                        if (!H1) {

                                //链表中无环

                                return False;

                        }

                         else

                         {

                                H1 = H1->next;

                                H2 = H2->next;

                        }

                }

        }

        //链表中无环

        return False;

}

和上一种实现代码一样,当函数返回 False 时,表明当前链表中无环;反之若返回 True,则表明该链表为有环链表。和第一种算法相比,本算法的时间复杂度为 O(n)。


 十八、双向循环链表(C语言)详解

我们知道,单链表通过首尾连接可以构成单向循环链表,如图 1 所示:

单向循环链表示意图

图 1 单向循环链表示意图

同样,双向链表也可以进行首尾连接,构成双向循环链表。如图 2 所示:

双向循环链表示意图

图 2 双向循环链表示意图

当问题中涉及到需要 "循环往复" 地遍历表中数据时,就需要使用双向循环链表。例如,前面章节我们对约瑟夫环问题进行了研究,其实约瑟夫环问题有多种玩法,每次顺时针报数后,下一轮可以逆时针报数,然后再顺时针......一直到剩下最后一个人。解决这个问题就需要使用双向循环链表结构。

双向循环链表的创建

创建双向循环链表,只需在创建完成双向链表的基础上,将其首尾节点进行双向连接即可

C 语言实现代码如下:

//创建双向循环链表

line* initLine(line * head){

        head=(line*)malloc(sizeof(line));

        head->prior=NULL;

        head->next=NULL;

        head->data=1;

        line * list=head;

        for (int i=2; i<=3; i++) {

                line * body=(line*)malloc(sizeof(line));

                body->prior=NULL;

                body->next=NULL;

                body->data=i;

                list->next=body;

                body->prior=list;

                list=list->next;

        }

        //通过以上代码,已经创建好双线链表,接下来将链表的首尾节点进行双向连接

        list->next=head;

        head->prior=list;

        return head;

}

通过向 main 函数中调用 initLine 函数,就可以成功创建一个存储有 {1,2,3} 数据的双向循环链表,其完整的 C 语言实现代码为:

#include <stdio.h>

#include <stdlib.h>

typedef struct line{

        struct line * prior;

        int data;

        struct line * next;

}line;

line* initLine(line * head);

void display(line * head);

int main() {

        line * head=NULL;

        head=initLine(head);

        display(head);

        return 0;

}

//创建双向循环链表

line* initLine(line * head){

        head=(line*)malloc(sizeof(line));

        head->prior=NULL;

        head->next=NULL;

        head->data=1;

        line * list=head;

        for (int i=2; i<=3; i++) {

                line * body=(line*)malloc(sizeof(line));

                body->prior=NULL;

                body->next=NULL;

                body->data=i;

                list->next=body;

                body->prior=list;

                 list=list->next;

        }

         //通过以上代码,已经创建好双线链表,接下来将链表的首尾节点进行双向连接

         list->next=head;

         head->prior=list;

         return head;

}

//输出链表的功能函数

void display(line * head){ l

        ine * temp=head;

        //由于是循环链表,所以当遍历指针temp指向的下一个节点是head时,证明此时已经循环至链表的最后一个节点

        while (temp->next!=head) {

                 if (temp->next==NULL) {

                         printf("%d\n",temp->data);

                 }else{

                         printf("%d->",temp->data);

                 }

                 temp=temp->next;

         }

         //输出循环链表中最后一个节点的值

         printf("%d",temp->data);

}

程序输出结果如下:

1->2->3

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

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

相关文章

Window端口占用处理

您好&#xff0c;我是码农飞哥&#xff08;wei158556&#xff09;&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f4aa;&#x1f3fb; 1. Python基础专栏&#xff0c;基础知识一网打尽&#xff0c;9.9元买不了吃亏&#xff0c;买不了上当。 Python从入门到精…

墨墨智库正式上线:开启您的AI智慧之旅

在这个由数据驱动的时代&#xff0c;AI技术正迅速改变我们的工作和生活方式。有没有想过一个平台可以为您提供所有AI相关资源的便捷访问&#xff1f;这就是「墨墨智库」的使命。我们非常高兴地宣布&#xff0c;经过精心准备和期待&#xff0c;「墨墨智库」现已正式上线&#xf…

Unity中Shader的_Time精度问题

文章目录 前言一、U方向上优化二、V方向上优化在这里插入图片描述 三、最终代码1、效果2、Shader 前言 在Unity的Shader中&#xff0c;使用了_Time来达到UV的流动效果&#xff0c;普遍会出现一个问题。我们的UV值会随着时间一直增加&#xff08;uv值增加了&#xff0c;但是因为…

1982-2022年GIMMS 标准化差异植被指数

GIMMS 标准化差异植被指数 1982-2022 PKU GIMMS 归一化植被指数数据集&#xff08;PKU GIMMS NDVI&#xff0c;版本 1.2&#xff09;提供了从 1982 年到 2022 年以半个月为间隔、分辨率为 1/12 的一致的全球 NDVI 数据。其主要目标是解决现有领域中普遍存在的关键不确定性。全…

AI教我学编程之C#关键字

AI教我学编程系列学习第三课 — C#关键字 前言重点先知关键字分类保留字上下文关键字 对话AI首遇波澜调整指令第一次第二次第三次直到我提出如下指令 人工智能&#xff1f;阶段总结 知识拓展1、Ecma和ISO是什么&#xff1f;2、System&#xff0c;dllhost.exe&#xff0c;taskmg…

基于神经网络的手写汉字提取与书写评分系统研究

相关源码和文档获取请私聊QQ:3106089953 论文目录结构 目 录 摘 要 I Abstract II 目 录 IV 第1章 绪论 1 1.1. 研究背景与意义 1 1.2. 国内外研究现状 2 1.2.1. 文本定位技术研究现状 2 1.2.2. 手写汉字识别研究现状 3 1.2.3. 汉字书写质量评价方法研究现状 4 1.3. 本文所做工…

SpringBoot配置Swagger2与Swagger3

swagger是什么&#xff1f; 在平时开发中&#xff0c;一个好的API文档可以减少大量的沟通成本&#xff0c;还可以帮助新加入项目的同事快速上手业务。大家都知道平时开发时&#xff0c;接口变化总是很多&#xff0c;有了变化就要去维护&#xff0c;也是一件比较头大的事情。尤…

IDEA TODO

今天记录一个 IDEA 工具的小技巧&#xff0c; TODO。比如下班前有一个小功能没完善好&#xff0c;此时可以在响应代码上加上 TODO 注解&#xff0c; //密码比对 // TODO 后期需要进行md5加密&#xff0c;然后再进行比对 password DigestUtils.md5DigestAsHex(password.getByt…

LeetCode 42:接雨水

一、题目描述 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图&#xff0c;计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 示例 1&#xff1a; 输入&#xff1a;height [0,1,0,2,1,0,1,3,2,1,2,1] 输出&#xff1a;6 解释&#xff1a;上面是由数组 [0,1,0,2,1,0,…

OpenSSL——升级

OpenSSL&#xff08;Open Secure Sockets Layer&#xff09;是一个开源的软件库&#xff0c;提供了SSL和TLS协议的实现&#xff0c;用于加密通信。它广泛用于安全连接&#xff0c;例如在网站上通过HTTPS协议进行安全的数据传输. 但是从openssl申请道德证书是不安全的。对于网站…

SpringBoot学习(二)-SpringBoot Web 开发

注&#xff1a;此为笔者学习狂神说SpringBoot的笔记&#xff0c;其中包含个人的笔记和理解&#xff0c;仅做学习笔记之用&#xff0c;更多详细资讯请出门左拐B站&#xff1a;狂神说!!! SpringBoot Web 开发 1、静态资源 1&#xff09;创建项目 2&#xff09;做项目都先测试一…

关于“Python”的核心知识点整理大全64

目录 20.2.15 确保项目的安全 settings.py 20.2.16 提交并推送修改 20.2.17 创建自定义错误页面 1. 创建自定义模板 500.html settings.py settings.py 注意 views.py 20.2.18 继续开发 往期快速传送门&#x1f446;&#xff08;在文章最后&#xff09;&#xff1a…