【算法通关指南:数据结构和算法篇】别再用指针写链表了!数组模拟单 / 双向链表,C++ 实战超丝滑

【算法通关指南:数据结构和算法篇】别再用指针写链表了!数组模拟单 / 双向链表,C++ 实战超丝滑
在这里插入图片描述
🔥小龙报:个人主页
🎬作者简介:C++研发,嵌入式,机器人方向学习者
❄️个人专栏:《算法通关指南》
永远相信美好的事情即将发生
在这里插入图片描述

文章目录


前言

本专栏聚焦算法题实战,系统讲解算法模块:以《c++编程》,《数据结构和算法》《基础算法》《算法实战》 等几个板块以题带点,讲解思路与代码实现,帮助大家快速提升代码能力
ps:本章节题目分两部分,比较基础笔者只附上代码供大家参考,其他的笔者会附上自己的思考和讲解,希望和大家一起努力见证自己的算法成长

一、链表的概念

1.1 链表的定义

线性表的链式存储就是链表。它是将元素存储在物理上任意的存储单元中,由于无法像顺序表⼀样通过下标保证数据元素之间的逻
辑关系,链式存储除了要保存数据元素外,还需额外维护数据元素之间的逻辑关系,这两部分信息合称结点(node)。即结点有两个域:保存数据元素的数据域和存储逻辑关系的指针域。

1.2 链表的分类

在这里插入图片描述

二、链表的模拟实现

2.1 单链表的模拟实现

2.1.1 定义-创建-初始化

(1) 两个足够大的数组,⼀个用来存数据,一个用来存下⼀个结点的位置
(2) 变量h ,充当头指针,表示头结点的位置
(3) 变量id,为新来的结点分位置

在这里插入图片描述
constint N =1e5+10;int h;//头指针int id;// 下⼀个元素分配的位置int e[N], ne[N];// 数据域和指针域// 下标0位置作为哨兵位// 其中ne数组全部初始化为0,其中ne[i] = 0就表⽰空指针,后续没有结点// 当然,也可以初始化为- 1作为空指针,看个⼈爱好

2.1.2 头插

voidpush_front(int x){ id++; e[id]= x;// 1. x 的右指针指向哨兵位的后继// 2. 哨兵位的右指针指向 ne[id]= ne[h]; ne[h]= id;}

时间复杂度:O(1)

2.1.3 遍历链表

//遍历链表voidprint(){for(int i = ne[0]; i; i = ne[i]){ cout << e[i]<< endl;}}

时间复杂度:O(1)

2.1.4 按值查找

策略一:遍历整个链表
intfind(int x){for(int i = ne[0]; i; i = ne[i]){if(e[i]== x)return i;}return0;}

时间复杂度:O(N)

策略二:使用哈希表优化
intfind(int x){return mp[x];}

时间复杂度:O(1)

2.1.5 在任意位置之后插入元素

voidinsert(int p,int x)// ⼀定要注意,这⾥的p是位置,不是元素{ id++; e[id]= x; ne[id]= ne[p];// x 指向p的后⾯ ne[p]= id;// p 指向x}

时间复杂度:O(1)

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

voiderase(int p){if(ne[p]){ mp[e[ne[p]]]=0;// 将p后⾯的元素从mp中删除 ne[p]= ne[ne[p]];//指向下⼀个元素的下⼀个元素}}

时间复杂度:O(1)

2.1.7 遗留问题

为什么不像顺序表⼀样,实现⼀个尾插、尾删、删除任意位置的元素等操作?
答:能实现,但是没必要。因为时间复杂度是O(N)级别的。
使用各种数据结构是⽅便我们去解决问题的,⽽不是添堵(增加时间复杂度)的

2.1.8 所有测试代码

#include<iostream> using namespace std;constint N =1e5+10;int h;//头指针int id;// 下⼀个元素分配的位置int e[N], ne[N];// 数据域和指针域int mp[N];// mp[i] 表⽰i在这个元素存放的位置//头插voidpush_front(int x){ id++; e[id]= x;// 1. x 的右指针指向哨兵位的后继// 2. 哨兵位的右指针指向 ne[id]= ne[h]; ne[h]= id; mp[x]= id;}//遍历链表voidprint(){for(int i = ne[0]; i; i = ne[i]){ cout << e[i]<<" ";} cout << endl << endl;}// 按值查找intfind(int x){return mp[x];}//在任意位置之后插⼊元素voidinsert(int p,int x)// ⼀定要注意,这⾥的p是位置,不是元素{ id++; e[id]= x; ne[id]= ne[p];// x 指向p的后⾯ ne[p]= id;// p 指向x}//删除存储位置为p后⾯的元素voiderase(int p){if(ne[p]){ mp[e[ne[p]]]=0;// 将p后⾯的元素从mp中删除 ne[p]= ne[ne[p]];//指向下⼀个元素的下⼀个元素}}intmain(){for(int i =1; i <=5; i++){push_front(i);print();}//cout << find(1) << endl;//cout << find(5) << endl;//cout << find(6) << endl;// insert(1, 10);// print();// insert(2, 100);// print();/*erase(2); print(); erase(4); print();*/return0;}

运行结果:

在这里插入图片描述

2.2 双向链表的模拟实现

2.2.1 实现方式

双向链表无非就是在单链表的基础上加上⼀个指向前驱的指针,那就 再来⼀个数组充当指向前驱的指针域 即可

2.2.2 定义

在这里插入图片描述
constint N =1e5+10;int h;// 头结点int id;// 下⼀个元素分配的位置int e[N];// 数据域int pre[N], ne[N];// 前后指针域

2.2.3 头插

//头插voidpush_front(int x){ id++; e[id]= x; pre[id]= h; ne[id]= ne[h]; pre[ne[h]]= id; ne[h]= id;}

时间复杂度:O(1)

2.2.4 遍历链表

直接无视prev 数组,与单链表的遍历方式一致。

voidprint(){for(int i = ne[0]; i; i = ne[i]){ cout << e[i]<<" ";} cout << endl << endl;}

时间复杂度:O(1)

2.2.5 按值查找

直接使用哈希表的思想进行优化

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

时间复杂度:O(1)

2.2.6 在任意位置之后插入元素

voidinsert_back(int p,int x){ id++; e[id]= x; pre[id]= p; ne[id]= ne[p];// 先让p的后继的左指针指向id// 再让p的右指针指向id pre[ne[p]]= id; ne[p]= id;}

时间复杂度:O(1)

2.2.7 在任意位置之前插入元素

voidinsert_front(int p,int x){ id++; e[id]= x; pre[id]= pre[p]; ne[id]= p;// 先让p的前驱的右指针指向id// 再让p的左指针指向id ne[pre[p]]= id; pre[p]= id;}

时间复杂度:O(1)

2.2.8 删除任意位置的元素

voiderase(int p){ ne[pre[p]]= ne[p]; pre[ne[p]]= pre[p];}

时间复杂度:O(1)

2.2.9 所有测试代码

#include<iostream> using namespace std;constint N =1e5+10;int h;// 头结点int id;// 下⼀个元素分配的位置int e[N];// 数据域int pre[N], ne[N];// 前后指针域int mp[N];//mp[i] 表⽰i在这个元素存放的位置//头插voidpush_front(int x){ id++; e[id]= x; pre[id]= h; ne[id]= ne[h]; pre[ne[h]]= id; ne[h]= id; mp[x]= id;}//遍历链表voidprint(){for(int i = ne[0]; i; i = ne[i]){ cout << e[i]<<" ";} cout << endl << endl;}// 按值查找intfind(int x){return mp[x];}// 在存储位置为p的元素后⾯,插⼊⼀个元素xvoidinsert_back(int p,int x){ id++; e[id]= x; pre[id]= p; ne[id]= ne[p];// 先让p的后继的左指针指向id// 再让p的右指针指向id pre[ne[p]]= id; ne[p]= id;}//在任意位置之前插入元素voidinsert_front(int p,int x){ id++; e[id]= x; pre[id]= pre[p]; ne[id]= p;// 先让p的前驱的右指针指向id// 再让p的左指针指向id ne[pre[p]]= id; pre[p]= id;}// 删除任意位置p的元素voiderase(int p){ ne[pre[p]]= ne[p]; pre[ne[p]]= pre[p];}intmain(){for(int i =1; i <=8; i++){push_front(i);print();}//cout << find(3) << endl;//cout << find(5) << endl;//cout << find(0) << endl;// insert_front(2, 22);// print();// insert_front(3, 33);// print();// insert_front(4, 44);// print();//erase(2);//print();//erase(4);//print();return0;}

运行结果:

在这里插入图片描述

2.3 循环链表的模拟实现

回看之前实现的带头单向链表。我们定义 表示空指针,但其实哨兵位就在 位置,所有的结构正好成环

三、总结与每日励志

本文介绍了链表这一重要的数据结构,重点讲解了单链表和双向链表的实现方法。作者通过数组模拟链表的方式,详细展示了链表的定义、初始化以及常见操作(头插、遍历、查找、插入和删除)的实现代码与时间复杂度分析。单链表使用两个数组分别存储数据和指针,双向链表则增加前驱指针数组。文章特别强调了哈希表在优化查找效率中的应用,并对比了不同操作的时间复杂度。通过实际代码示例,读者可以清晰理解链表的底层实现逻辑及其在算法竞赛中的应用优势。

在这里插入图片描述

Read more

【征文计划】AR健身教练:形随心动 - 基于Rokid CXR-M SDK的实践落地

【征文计划】AR健身教练:形随心动 - 基于Rokid CXR-M SDK的实践落地

一、项目背景与创意起源 在当今快节奏的都市生活中,健身已成为许多人保持健康的重要方式。然而,居家健身面临一个普遍痛点:缺乏专业指导,容易因动作不规范导致运动损伤,同时低头看手机或平板的体验也大大降低了健身的沉浸感和效率。 根据《2024年中国健身行业白皮书》显示,超过65%的居家健身用户表示"缺乏专业指导"是他们放弃健身的主要原因。而Rokid Glasses作为一款轻量级AR眼镜,其独特的"抬头即见"交互方式,为解决这一问题提供了绝佳的硬件基础。 "形随心动"创意的诞生源于一个简单但关键的观察:如果能将专业教练"投射"到用户视野中,实时指导动作,同时提供直观的数据反馈,那么居家健身体验将发生质的飞跃。通过Rokid CXR-M SDK的AI场景、自定义页面和提词器功能,我们能够实现这一愿景。 二、Rokid CXR-M SDK 相关 1. Rokid

By Ne0inhk
Flutter 三方库 eth_sig_util 的鸿蒙化适配指南 - 掌握以太坊加密签名核心技术、助力鸿蒙端 Web3 钱包与去中心化身份验证应用开发

Flutter 三方库 eth_sig_util 的鸿蒙化适配指南 - 掌握以太坊加密签名核心技术、助力鸿蒙端 Web3 钱包与去中心化身份验证应用开发

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 eth_sig_util 的鸿蒙化适配指南 - 掌握以太坊加密签名核心技术、助力鸿蒙端 Web3 钱包与去中心化身份验证应用开发 前言 在 OpenHarmony 鸿蒙应用的 Web3 浪潮中,安全性是应用生死存亡的关键。无论是构建非托管钱包、登录去中心化应用(dApp),还是执行 EIP-712 结构化数据的确认,都离不开严谨的以太坊签名与加密协议。eth_sig_util 作为一个专门针对以太坊签名习惯优化的 Dart 工具库,支持 personal_sign、signTypedData 以及公钥恢复等核心算法。本文将指导你如何在鸿蒙端集成 eth_sig_util,构建一套符合全球标准的加密验证体系。 一、原原理分析 / 概念介绍 1.

By Ne0inhk
宇树G1机器人强化学习训练完整实战教程

宇树G1机器人强化学习训练完整实战教程

0. 前言 人形机器人的运动控制一直是机器人领域的重要挑战,而强化学习为解决这一问题提供了强有力的工具。本教程将基于宇树G1人形机器人,从基础的强化学习环境搭建开始,逐步深入到高自由度模型的训练配置、奖励函数设计与优化,最终实现复杂动作的训练控制。作者看到一个很棒的系列,所以针对性的对文章内容进行了整理和二次理解,方便大家更好的阅读《不同自由度的宇树G1机器人强化学习训练配置及运行实战 + RSL-RL代码库问题修复》、《宇树G1机器人强化学习训练奖励函数代码架构 + 创建新的奖励函数(1)》、《RL指标分析与看板应用 — 宇树G1机器人高自由度模型强化学习训练实战(3)》、《调参解析 — 宇树G1机器人高自由度模型强化学习训练实战(4)》、《舞蹈训练?手撕奖励函数 — 宇树G1机器人高自由度模型强化学习训练实战(5)》。 1. 强化学习训练环境配置 1.1 基础环境搭建 宇树机器人的强化学习训练基于Isaac Gym物理仿真环境和RSL-RL强化学习框架。首先需要确保这两个核心组件正确安装和配置。 在开始训练之前,我们通过简单的命令来启动12自由度G1机器人的基础训练:

By Ne0inhk
GCC编译(6)静态库工具AR

GCC编译(6)静态库工具AR

GCC编译(6)静态库工具AR Author: Once Day Date: 2026年2月20日 一位热衷于Linux学习和开发的菜鸟,试图谱写一场冒险之旅,也许终点只是一场白日梦… 漫漫长路,有人对你微笑过嘛… 全系列文章可参考专栏: 编译构建工具链_Once-Day的博客-ZEEKLOG博客 参考文章:ar(1) - Linux manual page【Linux】ar命令:用于创建、修改和提取静态库(archive)-ZEEKLOG博客Linux命令学习手册-ar - 知乎Linux ar命令介绍 和常用示例 - Link_Z - 博客园 文章目录 * GCC编译(6)静态库工具AR * 1. AR工具概述 * 1.1 背景介绍 * 1.2 基础使用

By Ne0inhk