备战蓝桥杯 链表详解

链表概念

上一次我们用顺序存储实现了线性表,这次我们用链式存储结构实现的线性表就叫链表

链表每个节点包含数据本身和下一个节点和上一个节点的地址

链表的分类

单链表 双链表

带头链表 不带头链表

循环链表等等

我们竞赛一般都用的是带头链表

双向链表的特点是比较任意的找到前驱节点

循环链表的特点是从任意节点的位置开始都能遍历完整个链表

我们的动态实现链表是用new申请节点,delete释放节点

但是这种动态形式对时间消耗很大,所以我们竞赛中实现的是静态的链表

静态单链表的实现

我们用两个足够大的数组来实现静态链表

一个是elem数组用来存储每个节点的数据域

一个是next数组用来存储每个节点的指针域

一个变量h表示头节点的位置

一个变量id表示新加入节点的位置

静态单链表的头插

我们应该先把2连上,再进行1的连接,如果我们先进行1的连接,那么第一个节点就找不到了,2无法进行

所以我们要头插的时候,先让id++ 为新节点腾位置,再把数据的值给e[id]

然后我们实现2号连接,我们把ne[id]=ne[h]

最后我们实现1号连接,ne[h]=id;

我们的时间复杂度就是O(1)

代码:

void push_front(int x)
{e[++id] = x;ne[id] = ne[h];ne[h] = id;
}

遍历链表:

用for循环,int i先初始化为ne[h],i只要不等于0就继续进入循环,每次循环i都变成ne[i] 就完成了我们的遍历链表操作

代码

void Print()
{for (int i = ne[h]; i; i = ne[i]){cout << e[i] << " ";}cout << endl;
}

测试结果

查询节点 第一种方法就是遍历链表查询,返回节点的下标

int find(int x)
{//解法1遍历链表for (int i = ne[h]; i; i = ne[i]){if (e[i] == x)return i;}return 0;
}

查询节点的第二种方法,用空间替代时间,开一个mp数组,mp数组的下标就是我们链表的值,数组的值就是我们的链表节点的存储位置

相比于遍历的查询,我们mp数组的时间复杂度只是O(1),但是有两点局限性,1是数据的值不能太大,2是数据的值不能重复,不然就不知道存哪个下标了

int find(int x)
{return mp[x];
}

每次头插的时候 把新节点的存储位置更新到mp里

void push_front(int x)
{e[++id] = x;mp[x] = id;ne[id] = ne[h];ne[h] = id;
}

在任意位置之后 插入节点

如果我们先连接1号路线,那我们就找不到三这个节点了,所以我们应该先连接2号路线,也就是说,我们先让id++给新节点腾出位置,然后我们连接2号路线也就是让ne[id]=ne[p],再连接1号路线,也就是让ne[p]=id

void insert(int p, int x)
{//在存储位置为p的节点后面插入一个新节点id++;e[id] = x;mp[x] = id;ne[id] = ne[p];ne[p] = id;}

删除在任意位置之后的元素

如图,这种情况我们只要让1直接连接3 跳过2就行了,这时候虽然2还在我们的数组里面,但是遍历的时候不会遍历到他,也就相当于删除了这个节点了

也就是我们要删除2,就让ne[1]=ne[ne[1]]就行了,如果我们传p的话,就让ne[p]=ne[ne[p]]

当然,如果我们删除的节点是最后一个的下一个的话,就需要特殊判断一下,不然会存在bug

比如下图

我们要删除6节点的下一个节点的时候,ne[6]=ne[ne[6]],也就是ne[6]=ne[0],就会让我们的6节点和4节点再形成一个连接,破坏原有的结构

所以我们需要特判一下

void erase(int p)
{if (ne[p])//当p不是最后一个元素的时候{mp[e[ne[p]]] = 0;//清空标记ne[p] = ne[ne[p]];}
}

测试

我们单链表会了头插,查询存储位置,任意位置之后插入,任意位置之后删除这几个操作就够用了,其他的操作时间复杂度太高了我们基本用不到

静态单链表总代码

#include <iostream>
using namespace std;
const int N = 1e5 + 10;
int e[N], ne[N], h, id;
int mp[N];
void Print()
{for (int i = ne[h]; i; i = ne[i]){cout << e[i] << " ";}cout << endl;
}
void push_front(int x)
{e[++id] = x;mp[x] = id;ne[id] = ne[h];ne[h] = id;
}
int find(int x)
{解法1遍历链表//for (int i = ne[h]; i; i = ne[i])//{//	if (e[i] == x)//		return i;//}//return 0;return mp[x];
}
void insert(int p, int x)
{//在存储位置为p的节点后面插入一个新节点id++;e[id] = x;mp[x] = id;ne[id] = ne[p];ne[p] = id;}
void erase(int p)
{if (ne[p])//当p不是最后一个元素的时候{mp[e[ne[p]]] = 0;//清空标记ne[p] = ne[ne[p]];}
}
int main()
{for (int i = 1; i <= 5; i++){push_front(i);Print();}erase(2);Print();erase(3);Print();/*cout << find(1) << endl;cout << find(5) << endl;cout << find(6) << endl;*//*insert(3, 10);insert(5, 100);Print();*/
}

静态双链表的实现

下面我们来介绍一下双链表的实现,双链表无非就是在单链表的基础上增加了前驱指针,我们只需要多开一个足够大的数组来存储前面的元素的存储位置就行了

静态双链表的创建

#include <iostream>
using namespace std;
const int N = 1e5+10;
int e[N], ne[N], pre[N];
int id, h;

静态双链表的头插

a是哨兵位

和单链表一样,我们1号路线应该最后实现不然的话我们就找不到b这个节点了,自然也就连不上我们的链表了,我们先让id++为新节点腾出位置,然后e[id]=x 为了规范一下操作

我们首先先把新节点的pre指针和ne指针与相邻的两个节点连接

也就是ne[id]=ne[h] , pre[id] = h

接下来我们把b节点的前指针连接新节点,也就是3号路线

pre[ne[h]]=id

最后,我们修改哨兵位的ne指针,

ne[h]=id;

我们来实现一下代码

void push_front(int x)
{id++;e[id] = x;//先修改新来节点的左右指针ne[id] = ne[h];pre[id] = h;//再修改哨兵位下一个节点的左指针pre[ne[h]] = id;//最后修改哨兵位的右指针ne[h] = id;
}

测试头插

实现按值查找,我们还是用mp[N]空间代替时间

int find(int x)
{return mp[x];
}

在插入的时候更新mp数组

在任意位置之后插入元素

如图,我们想要在p这个存储位置后面插入一个元素,和头插差不多,id++,e[id]=x我们应该先更改新节点的左右指针,也就是pre[id] = p ne[id] = ne[p]

然后更改p右边的节点的左指针

pre[ne[p]]=id

最后更改p位置的右指针

ne[p]=id

代码实现

void insert_back(int p, int x)
{id++;e[id] = x;//修改新节点左右指针ne[id] = ne[p];pre[id] = p;//修改p后面节点的左指针pre[ne[p]] = id;//修改p的右指针ne[p] = id;
}

测试

在任意位置之前插入元素

删除任意位置的元素

循环链表

我们之前写的单链表其实就是循环链表,因为我们把最后一个节点的右指针写为0,其实就是头节点的下标

算法题练习:

1.排队顺序

第一行n是小朋友个数

第二行分别是第i个小朋友的下一个小朋友的编号,相当于我们链表的ne[N]的数组,而我们的数据域就是下标,我们只需要写一个遍历链表的代码就行了

第三行表示第一个小朋友的编号,我们遍历的起点

#include <iostream>
using namespace std;
const int N = 1e6+10;
int ne[N];
int n;
int main()
{cin >> n;for(int i = 1;i<=n;i++){cin >> ne[i];}int h;cin >> h;for(int i = h;i;i=ne[i]){cout << i << " ";} 
}

2.单向链表

3.队列安排

4.约瑟夫问题

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

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

相关文章

DeepSeek:性能强劲的开源模型

deepseek 全新系列模型 DeepSeek-V3 首个版本上线并同步开源。登录官网 chat.deepseek.com 即可与最新版 V3 模型对话。 性能对齐海外领军闭源模型​ DeepSeek-V3 为自研 MoE 模型&#xff0c;671B 参数&#xff0c;激活 37B&#xff0c;在 14.8T token 上进行了预训练。 论…

Redis Zset有序集合

个人主页&#xff1a;C忠实粉丝 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 C忠实粉丝 原创 Redis Zset有序集合 收录于专栏[redis] 本专栏旨在分享学习Redis的一点学习笔记&#xff0c;欢迎大家在评论区交流讨论&#x1f48c; 目录 概述 普通命令 ZAD…

Python中的可变对象与不可变对象;Python中的六大标准数据类型哪些属于可变对象,哪些属于不可变对象

Python中的可变对象与不可变对象&#xff1b;Python中的六大标准数据类型哪些属于可变对象&#xff0c;哪些属于不可变对象 Python中的可变对象与不可变对象一、Python的六大标准数据类型1. 数字类型 (Number)2. 字符串 (String)3. 列表 (List)4. 元组 (Tuple)5. 集合 (Set)6. …

Unity 2d描边基于SpriteRender,高性能的描边解决方案

目标 以Unity默认渲染管线为例&#xff0c;打造不需要图片内边距&#xff0c;描边平滑&#xff0c;高性能的描边解决方案 前言 在2d游戏中经常需要给2d对象添加描边&#xff0c;来突出强调2d对象 当你去网上查找2d描边shader&#xff0c;移植到项目里面&#xff0c;大概率会…

Oracle OCP考试常见问题之线上考试流程

首先要注意的是&#xff1a;虽然Oracle官方在国际上取消了获得OCP认证需要培训记录的要求&#xff0c;但在中国区&#xff0c;考生仍然需要参加Oracle的官方或者其合作伙伴组织的培训&#xff0c;并且由Oracle授权培训中心向Oracle提交学员培训记录。考生只有在完成培训并通过考…

基于海思soc的智能产品开发(camera sensor的两种接口)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 对于嵌入式开发设备来说&#xff0c;除了图像显示&#xff0c;图像输入也是很重要的一部分。说到图像输入&#xff0c;就不得不提到camera。目前ca…

Redis 笔记(二)-Redis 安装及测试

一、什么是 Redis 中文网站 Redis&#xff08;Remote Dictionary Server )&#xff0c;即远程字典服务&#xff0c;是一个开源的使用 ANSI C语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value&#xff0c;并提供多种语言的 API。 Redis 开源&#xff0c;遵循 BSD 基…

H2数据库在单元测试中的应用

H2数据库特征 用比较简洁的话来介绍h2数据库&#xff0c;就是一款轻量级的内存数据库&#xff0c;支持标准的SQL语法和JDBC API&#xff0c;工业领域中&#xff0c;一般会使用h2来进行单元测试。 这里贴一下h2数据库的主要特征 Very fast database engineOpen sourceWritten…

通俗易懂之线性回归时序预测PyTorch实践

线性回归&#xff08;Linear Regression&#xff09;是机器学习中最基本且广泛应用的算法之一。它不仅作为入门学习的经典案例&#xff0c;也是许多复杂模型的基础。本文将全面介绍线性回归的原理、应用&#xff0c;并通过一段PyTorch代码进行实践演示&#xff0c;帮助读者深入…

MATLAB深度学习实战文字识别

文章目录 前言视频演示效果1.DB文字定位环境配置安装教程与资源说明1.1 DB概述1.2 DB算法原理1.2.1 整体框架1.2.2 特征提取网络Resnet1.2.3 自适应阈值1.2.4 文字区域标注生成1.2.5 DB文字定位模型训练 2.CRNN文字识别2.1 CRNN概述2.2 CRNN原理2.2.1 CRNN网络架构实现2.2.2 CN…

和为0的四元组-蛮力枚举(C语言实现)

目录 一、问题描述 二、蛮力枚举思路 1.初始化&#xff1a; 2.遍历所有可能的四元组&#xff1a; 3.检查和&#xff1a; 4.避免重复&#xff1a; 5.更新计数器&#xff1a; 三、代码实现 四、运行结果 五、 算法复杂度分析 一、问题描述 给定一个整数数组 nums&…

SpringBoot日常:集成Kafka

文章目录 1、pom.xml文件2、application.yml3、生产者配置类4、消费者配置类5、消息订阅6、生产者发送消息7、测试发送消息 本章内容主要介绍如何在springboot项目对kafka进行整合&#xff0c;最终能达到的效果就是能够在项目中通过配置相关的kafka配置&#xff0c;就能进行消息…

【实用技能】如何使用 .NET C# 中的 Azure Key Vault 中的 PFX 证书对 PDF 文档进行签名

TX Text Control 是一款功能类似于 MS Word 的文字处理控件&#xff0c;包括文档创建、编辑、打印、邮件合并、格式转换、拆分合并、导入导出、批量生成等功能。广泛应用于企业文档管理&#xff0c;网站内容发布&#xff0c;电子病历中病案模板创建、病历书写、修改历史、连续打…

33.3K 的Freqtrade:开启加密货币自动化交易之旅

“ 如何更高效、智能地进行交易成为众多投资者关注的焦点。” Freqtrade 是一款用 Python 编写的免费开源加密货币交易机器人。它就像一位不知疲倦的智能交易助手&#xff0c;能够连接到众多主流加密货币交易所&#xff0c;如 Binance、Bitmart、Bybit 等&#xff08;支…

Mac M2基于MySQL 8.4.3搭建(伪)主从集群

前置准备工作 安装MySQL 8.4.3 参考博主之前的文档&#xff0c;在本地Mac安装好MySQL&#xff1a;Mac M2 Pro安装MySQL 8.4.3安装目录&#xff1a;/usr/local/mysql&#xff0c;安装好的MySQL都处于运行状态&#xff0c;需要先停止MySQL服务最快的方式&#xff1a;系统设置 …

事务的回滚与失效行为

创建一张测试表 AccountMapper public interface AccountMapper {Update("update account set balance #{balance} where username #{username}")int updateUserBalance(Param("username") String username, Param("balance") Integer bal…

【C语言】_字符数组与常量字符串

目录 1. 常量字符串的不可变性 2. 关于常量字符串的打印 3. 关于字符数组与常量字符串的内存分布 1. 常量字符串的不可变性 char arr[10] "abcdef";// 字符数组char* p2 arr;char* p3 "abcdef"; // 常量字符串 尝试对常量字符串进行修改&#xff…

【GUI-pyqt5】QCommandLinkButton类

1. 描述 命令链接的Windows Vista引入的新控件他的用途类似于单选按钮的用途&#xff0c;因为他用于在一组互斥选项之间进行选择命令链接按钮不应单独使用&#xff0c;而应作为向导和对话框中单选按钮替代选项外观通常类似于平面按钮的外观&#xff0c;但除了普通按钮文本外&a…

69.基于SpringBoot + Vue实现的前后端分离-家乡特色推荐系统(项目 + 论文PPT)

项目介绍 在Internet高速发展的今天&#xff0c;我们生活的各个领域都涉及到计算机的应用&#xff0c;其中包括家乡特色推荐的网络应用&#xff0c;在外国家乡特色推荐系统已经是很普遍的方式&#xff0c;不过国内的管理网站可能还处于起步阶段。家乡特色推荐系统采用java技术&…

HCIE-day10-ISIS

ISIS ISIS&#xff08;Intermediate System-to-Intermediate System&#xff09;中间系统到中间系统&#xff0c;属于IGP&#xff08;内部网关协议&#xff09;&#xff1b;是一种链路状态协议&#xff0c;使用最短路径优先SPF算法进行路由计算&#xff0c;与ospf协议有很多相…