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

【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.rhkb.cn/news/371147.html

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

相关文章

隔离级别-隔离级别中的锁协议、隔离级别类型、隔离级别的设置、隔离级别应用

一、引言 1、DBMS除了采用严格的两阶段封锁协议来保证并发事务的可串行化&#xff0c;实现事务的隔离性&#xff0c;也可允许用户选择一个可以保证应用程序正确执行并且能够使并发度最大的隔离性等级 2、通常用隔离级别来描述隔离性等级&#xff0c;以下将主要介绍ANSI 92标准…

【python技巧】parser传入参数

参考网址: https://lightning.ai/docs/pytorch/LTS/api/pytorch_lightning.utilities.argparse.html#pytorch_lightning.utilities.argparse.add_argparse_args 1. 简单传入参数. parse_known_args()方法的作用就是把不在预设属性里的参数也返回,比如下面这个例子, 执行pytho…

算法的空间复杂度(C语言)

1.空间复杂度的定义 算法在临时占用储存空间大小的量度&#xff08;就是完成这个算法所额外开辟的空间&#xff09;&#xff0c;空间复杂度也使用大O渐进表示法来表示 注&#xff1a; 函数在运行时所需要的栈空间(储存参数&#xff0c;局部变量&#xff0c;一些寄存器信息等)…

vue.js微商城后台管理系统

一.需要运行的效果 20240701-231456 二.代码&#xff08;解析&#xff09; 首先&#xff0c;为项目添加依赖&#xff1a; yarn add element-plus --save yarn add vue-router4 --save 新建一个项目包&#xff0c;然后命名为商品管理&#xff0c;在components中新建几个vue文件…

PLC电源模块

PM电源模块 为CPU信号模块及 其他的扩展设备、其他用电设备&#xff08;如传感器&#xff09;提供工作供电 接线和开关 状态显示 灯的闪烁示意看手册 PS电源模块 为CPU信号模块及其他的扩展设备提供工作供电。PS(System Power Supply) 外形与PM电源模块类似&#xff0c;状…

STM32-USART

本内容基于江协科技STM32视频学习之后整理而得。 文章目录 1. 串口通信协议1.1 通信接口1.2 串口通信1.3 硬件电路1.4 电平标准1.5 串口参数及时序1.6 串口时序 2. USART串口通信2.1 USART简介2.2 USART框图2.3 USART基本结构2.4 数据帧2.5 数据帧-配置停止位2.6 起始位侦测2.…

【flutter问题记录】 无效的源发行版:17

问题描述 在看开源项目的时候&#xff0c;clone下来后一直编译失败&#xff0c;提示&#xff1a;无效的源发行版:17&#xff0c;看描述大概是jdk的版本问题&#xff0c;但是在Android studio各种指定都无用&#xff0c;网上资料也没有flutter项目的解决方案&#xff0c;最后在…

【每日一练】python三目运算符的用法

""" 三目运算符与基础运算的对比 """ a 1 b 2#1.基础if运算判断写法&#xff1a; if a > b:print("基础判断输出&#xff1a;a大于b") else:print("基础判断输出&#xff1a; a不大于b")#2.三目运算法判断&#xff1a;…

转盘输入法-键盘加鼠标版本

序 转盘输入法&#xff0c;给你的聊天加点新意。它不用常见的九宫格或全键盘&#xff0c;而是把字母摆在圆盘上&#xff0c;一滑一滑&#xff0c;字就出来了&#xff0c;新鲜又直接。 键盘加鼠标版本GIF演示 演示软件下载 转盘输入法PC演示版本EXE下载https://download.csdn…

机械键盘有哪些分类

机械键盘是一种比传统的薄膜键盘更耐用、更快捷、更具有手感的键盘。它的键帽和按键是独立的&#xff0c;能够提供更好的反应速度和操作感。机械键盘在现代化生活中得到了广泛的应用。根据其特性和使用场景&#xff0c;机械键盘可以分为以下几类&#xff1a; 1.轴体分类 机械…

妈妈带女儿美在心里

在这个充满温情与惊喜的午后&#xff0c;阳光温柔地洒落在每一个角落&#xff0c;仿佛连空气弥漫着幸福的味道。就在这样一个平凡的时刻&#xff0c;一段关于爱与成长的温馨画面&#xff0c;悄然在网络上绽放&#xff0c;引爆了无数人的心弦——#奚梦瑶2岁女儿身高#&#xff0c…

STM32远程烧录程序

目录 简介 不同的程序下载方式 ICP&#xff1a;In-Circuit Programming ISP&#xff1a;In-System Programing IAP&#xff1a;In-Application Programming BootLoader Bootloader 是什么&#xff1f; STM32的启动方式 存储器组织 存储器映像 嵌入式SRAM 嵌入式FL…

安卓备忘录App开发

安卓备忘录APP开发,文章末尾有源码和apk安装包 目标用户: 普通安卓手机用户,需要一个简单易用的备忘录App来记录和管理日常事务。 主要功能: 用户注册: 用户可以创建一个账号,输入用户名和密码。 用户登录: 用户可以通过用户名和密码登录到应用。 用户信息存储: 用户名和…

WAWA鱼曲折的大学四年回忆录

声明&#xff1a;本文内容纯属个人主观臆断&#xff0c;如与事实不符&#xff0c;请参考事实 前言&#xff1a; 早想写一下大学四年的总结了&#xff0c;但总是感觉无从下手&#xff0c;不知道从哪里开始写&#xff0c;通过这篇文章主要想做一个记录&#xff0c;并从现在的认…

Python爬取股票信息-并进行数据可视化分析,绘股票成交量柱状图

为了使用Python爬取股票信息并进行数据可视化分析&#xff0c;我们可以使用几个流行的库&#xff1a;requests 用于网络请求&#xff0c;pandas 用于数据处理&#xff0c;以及 matplotlib 或 seaborn 用于数据可视化。 步骤 1: 安装必要的库 首先&#xff0c;确保安装了以下P…

Linux--V4L2摄像头驱动框架及UVC浅析

一、前言 对于一个usb摄像头&#xff0c;它的内核驱动源码位于/drivers/media/usb/uvc/ 核心层&#xff1a;V4L2_dev.c文件 硬件相关层&#xff1a; uvc_driver.c文件 本篇记录基于对6.8.8.8内核下vivid-core.c文件&#xff08;虚拟视频驱动程序&#xff09;的分析&#xff…

【Unity数据交互】如何Unity中读取Ecxel中的数据

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;元宇宙-秩沅 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 秩沅 原创 &#x1f468;‍&#x1f4bb; 专栏交流&#x1f9e7;&…

xxl-job集成SpringBoot

安装xxl-job客户端一般有很多方式&#xff0c;我这里给大家提供两种安装方式&#xff0c;包含里面的各项配置等等。 前期需要准备好MySQL数据库。复制SQL到数据库里面。 # # XXL-JOB v2.4.2-SNAPSHOT # Copyright (c) 2015-present, xuxueli.CREATE database if NOT EXISTS x…

自动控制:前馈控制

自动控制&#xff1a;前馈控制 前馈控制是一种在控制系统中通过预先计算和调整输入来应对已知扰动或变化的方法。相比于反馈控制&#xff0c;前馈控制能够更快速地响应系统的变化&#xff0c;因为它不依赖于系统输出的反馈信号。前馈控制的应用在工业过程中尤为广泛&#xff0…

计算机网络--网络层

一、网络层的服务和功能 网络层主要为应用层提供端对端的数据传输服务 网络层接受运输层的报文段&#xff0c;添加自己的首部&#xff0c;形成网络层分组。分组是网络层的传输单元。网络层分组在各个站点的网络层之间传输&#xff0c;最终到达接收方的网络层。接收方网络层将运…