【数据结构】链表带环问题分析及顺序表链表对比分析

【C语言】链表带环问题分析及顺序表链表对比分析

🔥个人主页大白的编程日记

🔥专栏C语言学习之路


文章目录

  • 【C语言】链表带环问题分析及顺序表链表对比分析
    • 前言
    • 一.顺序表和链表对比
      • 1.1顺序表和链表的区别
      • 1.2缓存利用率(缓存命中率)
    • 二.链表的带环问题
      • 2.1快慢指针
      • 2.2证明快慢指针相遇问题
      • 2.3快指针的步长
      • 2.4环的入口
    • 后言

前言

哈喽,各位小伙伴大家好!由于考试周很久没有更新博客了。今天给大家带来的是链表的带环问题和顺序表链表的对比分析。话不多说,进入正题。向大厂冲锋!

一.顺序表和链表对比

1.1顺序表和链表的区别

顺序表和链表是两种不同的数据结构。他们各有各的优劣。我们就来对比分析一下他们的区别。我们这里用带头双向循环链表和顺序表做对比。

  • 存储空间
    顺序表:物理上是连续的。
    链表:因为链表是由节点组成,每个节点由指针连接。 所以在逻辑上是连续的,但每个节点都是malloc动态开辟的,在物理空间上不一定连续。
  • 随机访问
    顺序表:顺序表可以通过下标来进行随机访问。
    链表:链表不支持随机访问,只能从头节点开始遍历寻找节点。
  • 任意位置插入删除
    顺序表:如果不是尾插尾删,需要挪动数据。
    链表:链表由节点组成,插入或删除只需要修改前后节点的指针指向即可。
  • 扩容
    顺序表:空间不够需要扩容。
    扩容realloc本身会有消耗且异地扩容消耗不小,2倍扩容可能存在空间浪费。
    链表:按需申请释放,需要一个申请一个,不存在扩容,不会浪费空间。
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
int main()
{int* p = (int*)malloc(4);printf("%p\n", p);int* p1 = (int*)realloc(p, 40);printf("%p", p1);
}

异地扩容:
只要空间大一点,基本都是异地扩容。
原地扩容:

  • 应用场景

顺序表和链表的优劣是互补的。
顺序表适合随机访问,不适合中间位置的插入删除。
链表适合任意位置的插入删除,但无法随机访问。
所以如果经常随机访问,但只需要尾插尾删就选择顺序表。
如果不经常随机访问,在中间位置插入删除就选择链表。
具体根据他们的优劣进行选择。

1.2缓存利用率(缓存命中率)

顺序表和链表的区别还有一个就是
顺序表的缓存命中率高。
链表的缓存命中率低。

为什么呢?什么是缓存命中率呢?

  • 内存和硬盘

这是我们计算机的内部的存储结构。
主存也就是我们的内存和硬盘的区别就是

内存的存储空间更小,通常为8G和16G,但速度快。需要带电存储
硬盘存储空间更大,速度慢,但不需要带电存储。
他们的本质是带不带电。

例如:

如果我正在写一份ppt,因为硬盘的速度慢,所以是存在内存中的,如果我这时电脑突然没电关机。重新开机后,我的ppt就不见了。因为我没有另存到硬盘中。
只用当我们另存到硬盘中才存在。

  • 寄存器和三级缓存
    那既然已经有内存,内存的速度也还行,为什么还有寄存器和三级缓存呢?
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
int main()
{int i = 0;int ret = i++;
}

以这段代码为例:

i存在内存中,也就是main函数的栈帧里。i++的执行过程是这样的:
先把i放在eax寄存器中,
对eax++,
把eax寄存器放i的内存位置

那为什么要这样做呢?
因为CPU和内存不同频,CPU跑的太快了。
如果直接访问内存数据进行++,因为内存太慢了。
他宁愿把内存中数据加载到寄存器中,CPU在寄存器执行指令,再把运行结果返回内存。

一般来说,CPU不会直接访问内存

  • 寄存器
    如果数据比较小(4或8字节)就会把数据加载到寄存器。

  • 缓存
    如果数据比较大就加载到缓存中。

缓存命中:如果要访问的数据在缓存,叫缓存命中,直接访问。
缓存不命中,如果要访问的数据不在缓存,叫缓存不命中,先把数据加载到缓存中,再访问。

  • 缓存的加载
    如果你要加载4个字节到缓存,通常会加载一长段空间到缓存中。而不只是4个字节。为什么呢?

把内存看作学校,缓存看作大巴,CPU看作度假村。
现在学校安排大巴把学生(数据)送到度假村去。

所以顺序表的缓存命中率高,
链表的缓存命中率低,而且会造成缓存污染。
如果大家想多了解缓存的话可以看这篇文章
与程序员相关的CPU缓存知识

二.链表的带环问题

链表带环是链表中的经典问题,值得我们深入学习。解决带环问题通常使用快慢指针相遇解决。但是你如何证明快慢指针一定相遇,以及快指针的步长不同会怎样呢?接下来,小编带大家一一探讨。

2.1快慢指针

  • 题目
    环形链表

  • 思路
    创建一个快指针和一个慢指针,快指针一次走两步,慢指针一次走一步。
    如果是链表带环,快慢指针最终会相遇。不带环,则快指针走到尾。
  • 代码实现
 typedef struct ListNode ListNode; 
bool hasCycle(struct ListNode *head) 
{ListNode*slow,*fast;slow=fast=head;while(fast&&fast->next){slow=slow->next;//慢指针走一步fast=fast->next->next;//快指针走两步if(fast==slow)//快慢指针相遇{return true;}}return false;//不带环
}

2.2证明快慢指针相遇问题

那如何证明题目一定会相遇呢?

当慢指针入环时,快指针与慢指针相差N个节点。
由于快指针每次走两步,慢指针走一步。
每次移动快指针都会与慢指针的距离缩小一个节点。
当他们的距离节点缩小为0时,就会相遇。
所以快慢指针一定能够相遇。

2.3快指针的步长

那快指针是不是只能走一步呢?如果快指针走3,4,5…N步还一定能相遇吗?

  • 步长为3时
    证明结果如下

我们用快慢指针步长的关系列出等式,反推证明N为奇数和C为偶数的情况不会出现,从而得出结论步长为3时一定能相遇。

  • 验证

  • 步长为3,4,5…N
    这些情况和前面的推导证明过程相似,大家有兴趣可以自己深入探究。

2.4环的入口

  • 题目
    环形链表二

  • 思路
    创建一个快指针和一个慢指针,快指针一次走两步,慢指针一次走一步。
    如果是链表带环,快慢指针最终会相遇。
    一个指针相遇点开始走,一个指针从头节点开始走,每次两个指针都走一步。
    当两个指针相遇时,相遇节点就是入环节点。
    不带环,则快指针走到尾。

  • 代码实现

 typedef struct ListNode ListNode ;
struct ListNode *detectCycle(struct ListNode *head) 
{ListNode*slow,*fast;slow=fast=head;while(fast&&fast->next){fast=fast->next->next;slow=slow->next;if(slow==fast){ListNode* pcur=slow;while(pcur!=head){pcur=pcur->next;head=head->next;}return pcur;}}return NULL;
}
  • 证明

具体证明过程如下:

  • 验证
    -在这里插入图片描述

所以根据推导我们得出只要再相遇后,一个head指针从头节点出发,一个pcur节点从相遇点出发,等他们相遇时,相遇点就是入环点。

后言

这就是链表的带环问题和顺序表链表的对比。这些都是我们数据结构学习时的重要内容。大家一定要好好掌握。今天就分享到这里,咱们下期见!拜拜~
在这里插入图片描述

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

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

相关文章

Linux基础: 二. Linux的目录和文件

文章目录 二. Linux的目录和文件1.1 目录概要1.2 目录详细说明 二. Linux的目录和文件 1.1 目录概要 command&#xff1a;ls / Linux的文件系统像一棵树一样&#xff0c;树干是根目录&#xff08;/&#xff09;&#xff0c;树枝是子目录&#xff0c;树叶是文件&#xff1b; …

亚信安全发布2024年6月威胁态势,高危漏洞猛增60%

近日&#xff0c;亚信安全正式发布《2024年6月威胁态势报告》&#xff08;以下简称“报告”&#xff09;&#xff0c;报告显示&#xff0c;6月份新增信息安全漏洞 1794个&#xff0c;高危漏洞激增60%&#xff0c;涉及0day漏洞占67.67%&#xff1b;监测发现当前较活跃的勒索病毒…

应用案例 | 基于物联网工控屏的工业离心机设备监控系统

案例概况 客户&#xff1a;博鲁班特&#xff08;BROADBENT&#xff09; 应用产品&#xff1a;宏集物联网工控屏 应用场景&#xff1a;离心机设备监控系统 一、前言 在现代工业生产中&#xff0c;离心机作为关键的分离设备&#xff0c;在生产过程中扮演着至关重要的角色。随…

谷粒商城学习笔记-17-快速开发-逆向工程搭建使用

文章目录 一&#xff0c;克隆人人开源的逆向工程代码二&#xff0c;把逆向工程集成到谷粒商城的后台工程三&#xff0c;以商品服务为例&#xff0c;使用逆向工程生成代码1&#xff0c;修改逆向工程的配置2&#xff0c;以Debug模式启动逆向工程3&#xff0c;使用逆向工程生成代码…

基于B/S模式和Java技术的生鲜交易系统

你好呀&#xff0c;我是计算机学姐码农小野&#xff01;如果有相关需求&#xff0c;可以私信联系我。 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;B/S模式、Java技术 工具&#xff1a;Visual Studio、MySQL数据库开发工具 系统展示 首页 用户注册…

【Java】详解String类中的各种方法

创建字符串 常见的创建字符串的三种方式&#xff1a; // 方式一 String str "hello world"; // 方式二 String str2 new String("hello world"); // 方式三 char[] array {a, b, c}; String str3 new String(array); "hello" 这样的字符串字…

Halcon 产品周围缺口检测

*读取一张图像read_image (Image, 原图.jpg)*获取图像大小get_image_size(Image, Width, Height)*关闭已经打开的窗口dev_close_window ()*打开新窗口dev_open_window(0, 0, Width, Height, black, WindowHandle) //打开指定大小的窗口*对图像进行阈值操作threshold (Image, R…

【链表】【双指针】1、合并两个有序链表+2、分隔链表+3、删除链表的倒数第N个结点+4、链表的中间结点+5、合并两个链表

3道中等2道简单 数组和字符串打算告一段落&#xff0c;正好最近做的几乎都是双指针&#xff0c;所以今天做链表&#xff01; 1、合并两个有序链表&#xff08;难度&#xff1a;简单&#xff09; 该题对应力扣网址 AC代码 思路简单 /*** Definition for singly-linked list.…

万和day01代码分析

将了数据库的多表之间的操作&#xff0c;实际应用到JDBC中去。 一共五张表&#xff0c; info存储的是具体的信息&#xff0c;edu job role 和info都是多对一的关系。 采用的是Java FX&#xff0c;界面采用xml去编写。 项目理解一 在JavaFX中&#xff0c;ObservableList 是一个…

SCI一区TOP|准随机分形搜索算法(QRFS)原理及实现【免费获取Matlab代码】

目录 1.背景2.算法原理2.1算法思想2.2算法过程 3.结果展示4.参考文献5.代码获取 1.背景 2024年&#xff0c;LA Beltran受到分形几何、低差异序列启发&#xff0c;提出了准随机分形搜索算法&#xff08;Quasi-random Fractal Search, QRFS&#xff09;。 2.算法原理 2.1算法思…

【网络安全】实验三(基于Windows部署CA)

一、配置环境 打开两台虚拟机&#xff0c;并参照下图&#xff0c;搭建网络拓扑环境&#xff0c;要求两台虚拟的IP地址要按照图中的标识进行设置&#xff0c;并根据搭建完成情况&#xff0c;勾选对应选项。注&#xff1a;此处的学号本人学号的最后两位数字&#xff0c;1学号100…

Linux 搭建 Kafka 环境 - 详细教程

目录 一. Kafka介绍 1. 应用场景 2. 版本对比 二. Kafka安装 1. 前置环境 &#xff08;1&#xff09;安装JDK 2. 软件安装 &#xff08;3&#xff09;环境变量配置 &#xff08;3&#xff09;服务启动 三. Console测试 基础命令 &#xff08;1&#xff09;列出Kafk…

上网监控软件有哪些?3款实力出众的上网监控软件

为什么需要上网监控软件&#xff1f; 据说&#xff0c;99%的员工上班都会摸鱼&#xff0c;1%的员工上班会窃取公司信息。 所以&#xff0c;因此&#xff0c;监控员工的上网行为是很有必要滴。 总结下来&#xff0c;上网监控软件的作用是&#xff1a; 1.提高生产力&#xff1…

前端位置布局汇总

HTML中脱离文档流的元素有&#xff1a; position: absolute - 元素相对于最近的已定位&#xff08;非 static&#xff09;祖先元素定位。 position: fixed - 元素相对于浏览器窗口定位。 float: left 或 float: right - 元素向左或向右浮动&#xff0c;周围的内容会环绕它。 …

shark云原生-日志体系-filebeat高级配置(适用于生产)-更新中

文章目录 1. filebeat.inputs 静态日志收集器2. filebeat.autodiscover 自动发现2.1. autodiscover 和 inputs2.2. 如何配置生效2.3. Providers 提供者2.4. Providers kubernetes2.5. 配置 templates2.5.1. kubernetes 自动发现事件中的变量字段2.5.2 配置 templates 2.6. 基于…

IDEA发疯导致maven下载回来的jar不完整zip END header not found

IDEA发疯导致maven下载回来的jar不完整zip END header not found 具体报错 java: 读取D:\mavenRepository\com\alibaba\druid-spring-boot-starter\1.2.23\druid-spring-boot-starter-1.2.23.jar时出错; zip END header not foundjava: java.lang.RuntimeException: java.io.…

2024 JuniorCryptCTF reppc 部分wp

Random cipher 文本编辑器打开附件 比较简单。脚本 Mutated Caesar 文本编辑器打开附件 比较简单。脚本 Pizza 附件拖入dnSpy 比较简单。脚本 l33t Leet&#xff0c;又称黑客语&#xff0c;是指一种发源于欧美地区的BBS、线上游戏和黑客社群所使用的文字书写方式&#xff0c;通…

Linux:进程终止和进程替换

Linux&#xff1a;Linux&#xff1a;进程终止和进程替换 一、进程终止1.1 进程退出场景和创建退出方式 1.2 exit 和 _exit区别二、进程程序替换2.1 进程替换函数2.2 函数解释及命名解释函数解释命名解释 2.3 单进程程序替换&#xff08;无子进程&#xff09;2.3.1 带l函数进程替…

买的Google账号登录,修改辅助邮箱收不到验证码?可能是个简单的错误

这篇文章分享一个案例&#xff0c;购买了谷歌账号以后如何修改辅助邮箱&#xff0c;修改辅助邮箱的一些要点&#xff0c;以及常见的一个错误。 一、案例回放 这个朋友昨天在我的一个视频下面留言说买了谷歌账号以后&#xff0c;想修改辅助邮箱地址&#xff0c;但是输入了辅助…

【WebRTC实现点对点视频通话】

介绍 WebRTC (Web Real-Time Communications) 是一个实时通讯技术&#xff0c;也是实时音视频技术的标准和框架。简单来说WebRTC是一个集大成的实时音视频技术集&#xff0c;包含了各种客户端api、音视频编/解码lib、流媒体传输协议、回声消除、安全传输等。对于开发者来说可以…