【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表)

💐 🌸 🌷 🍀 🌹 🌻 🌺 🍁 🍃 🍂 🌿 🍄🍝 🍛 🍤
📃个人主页 :阿然成长日记 👈点击可跳转
📆 个人专栏: 🔹数据结构与算法🔹C语言进阶
🚩 不能则学,不知则问,耻于问人,决无长进
🍭 🍯 🍎 🍏 🍊 🍋 🍒 🍇 🍉 🍓 🍑 🍈 🍌 🍐 🍍
文章目录
- 一、链表的概念
- 二、特点
- 三、链表的分类
- 四、单向链表的结构体
- 命名规范:
- 二级指针
- ❗️注意事项
- 五、函数实现
- 1.单链表的打印
- 2.单链表的头插
- 3.单链表的尾插
- 4.单链表的头删
- 5.单链表尾删
- 6.在pos位置之前插入x
- 7.在pos位置之后插入x
- 8.删除pos位置 节点
- 9.删除pos位置之后的节点
- 10.单链表的查找
前言
🎸小伙伴们,又见面了🌻 🌺 🍁 🍃 前面我们学习啦顺序表,其实顺序表的时间复杂度是很高的,尤其是在插入,删除等问题上,需要移动整个数组,十分麻烦费时。有没有更好的办法呢????当然有呀,就是链表,也是本篇博客要详细讲解的。
一、链表的概念
链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表 中的指针链接次序实现的 。
链表就像是一列火车,链表中的每一个节点,就像是火车的一节节车厢。

图1.1
图1.2
| 上面两幅图片生动地解释了链表的物理结构。想必看到这里已经对链表有了初步的认识。 |
二、特点
1️⃣ 链式结构在逻辑上是连续的,但在物理层上不一定连续。
2️⃣节点一般都是从堆上申请出来的一块空间。
3️⃣从堆上申请的空间,按照它的规则来进行分配,两次申请的空间,不一定连续。
三、链表的分类
1.单向或者双向
2. 带头或者不带头
3. 循环或者非循环
四、单向链表的结构体
❌误区:以下这种结构体定义会报错,那么是为什么呢?
typedef int SLTDataType;
typedef struct SListNode
{SLTDataType data;SLTNode* next;//错误}SLTNode;
我们的typedef关键字给结构体重新命名为SLTNode,但是他是在结构体最后才生效,如果现在就在结构体中使用新命名,那么就会找不到。
👍正解是:
typedef int SLTDataType;
typedef struct SListNode
{SLTDataType data;struct SListNode* next;}SLTNode;
1.node:是存储的数据;
2.next 的类型是一个节点型的指针变量,它保存的是下一个节点的地址,即指向下一个节点
命名规范:
当我们在给结构体命名或者是函数的命名我们都应该使用用英文或者英文的简写来进行命名这样有利于人们的理解。例如单链表英文名:single List table,所以我给节点命名为SLTNode.
二级指针
在下面的学习中,会使用二级指针,不太清楚的小伙伴,可以去看我的📋C进阶专栏中的👉高级指针一篇
❗️注意事项
我们现在定义的头指针在函数结束之后都会销毁,因为它存在栈上。我们的每一个节点是使用动态内存函数在堆上进行开辟如果不进行free释放那么它会持续保存到程序结束。
五、函数实现
1.单链表的打印
//打印单链表
void PrintSlistTable(SLTNode* phead)
{SLTNode* cur = phead;while (cur != NULL){printf("%d->", cur->data);cur = cur->next;}printf("NULL");
}
2.单链表的头插
头插思路分析:

头插代码
//头插
void SLTPusFront(SLTNode** pphead, SLTDataType x)//放入新插入节点
{SLTNode* newnode = CreatNode(x);newnode->next = *pphead;*pphead = newnode;
}
这里有很多小伙伴都不知道为什么使用了二级指针。因为在传参时我们使用的是结构体地址传参,这样能节省空间,提高效率,传入的是一级指针phead的地址,所以我们需要使用二级指针pphead来接收。
3.单链表的尾插
尾插思路分析:

尾插代码:
//尾插
void SLTPusBack(SLTNode** pphead, SLTDataType x)
{SLTNode* newnode = CreatNode(x);if (*pphead == NULL){//改变的结构体的指针,所以要用二级指针 *pphead = newnode;}else{SLTNode* tail = *pphead;while (tail->next != NULL){tail = tail->next;}//改变结构体,用结构体指针即可 tail->next = newnode;}
}
4.单链表的头删
思路:
一个节点和多个节点处理方式相同

代码:
//头删
void PopFront(SLTNode** pphead)
{assert(*pphead);SLTNode* cur = (*pphead)->next;free(*pphead);*pphead = cur;
}
1️⃣ 定义一个cur临时指针用来指向头节点的下一个节点.SLTNode* cur = (*pphead)->next;
2️⃣ 释放 *pphead即(删除第一个节点)free(*pphead);
3️⃣ 在将 *pphead指向第二节点*pphead = cur;
5.单链表尾删
思路:
1.如果没有节点,则直接释放头指针所指向的内容
2.

代码:
//尾插
void SLTPusBack(SLTNode** pphead, SLTDataType x)
{SLTNode* newnode = CreatNode(x);if (*pphead == NULL){//改变的结构体的指针,所以要用二级指针 *pphead = newnode;}else{SLTNode* tail = *pphead;while (tail->next != NULL){tail = tail->next;}//改变结构体,用结构体指针即可 tail->next = newnode;}
}
6.在pos位置之前插入x
思路:

代码:
//在pos位置之前插入x
void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{assert(*pphead);assert(pos);if (*pphead == pos){//头插;SLTPusFront(pphead, x);}else{//定义一个临时指针cur指向头指针,为了从头开始遍历各个节点找pos,而不会改变头指针pphead的指向位置。SLTNode* cur = *pphead;while (cur->next != pos){cur = cur->next;}SLTNode* newNode = CreatNode(x);newNode->next = cur->next;cur->next = newNode;free(cur);cur = NULL;}
}
定义一个临时指针cur指向头指针,用来从头开始遍历各个节点找pos,
头指针pphead的指向位置不能变,不然就找不到头了。
7.在pos位置之后插入x
思路:

代码:
在pos位置之后插入x
void SLTInsertAfter(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{assert(*pphead);SLTNode* newNode = CreatNode(x);newNode->next = pos->data;pos->next = newNode;
}
8.删除pos位置 节点
思路:

代码:
//删除POS位置
void SLTErase(SLTNode** pphead, SLTNode* pos)
{assert(*pphead);assert(pos);if (*pphead == pos){//头删SLTPopFront(pphead);}else{SLTNode* cur = *pphead;while (cur->next == pos){cur = cur->next;}pos->next = cur->next;free(pos);}
}
9.删除pos位置之后的节点
思路:

代码:
//删除POS之后的位置
void SLTEraseAfter(SLTNode** pphead, SLTNode* pos)
{assert(pphead);assert(pos);assert(pos->next);SLTNode* cur = pos->next->next;free(pos->next);pos->next = cur;}
10.单链表的查找
代码:
//单链表的查找 SLTNode* SLTSrech(SLTNode** pphead, SLTDataType x)
{SLTNode* cur = *pphead;while (cur->next!= NULL){if (cur->data == x)return cur;cur = cur->next;}return cur;
}
相关文章:
【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表)
💐 🌸 🌷 🍀 🌹 🌻 🌺 🍁 🍃 🍂 🌿 🍄🍝 🍛 🍤 📃个人主页 :阿然成长日记 …...
ubuntu开机自启动
ubuntu开机自启动 1、建一个test.sh脚本,并写入 #!/bin/sh gnome-terminal -x bash -c ‘cd /home/文件路径/;python3 main.py’ exit 0 2、:wq!保存 3、创建rc-local.service文件(sudo vim /etc/systemd/system/rc-local.service)…...
Git将其他分支合并至主分支
主要思想: 把分支代码合并到master,合给谁,就先切换到谁的分支 1. 当前分支是dev,开发完成后,需要合并到master分支 先把该提交的提交,需要push的push完成后,再切换分支。 否则也会告诉你要提交…...
Python+request+pytest 接口自动化测试框架入门(与unittest的比较)
1. Pythonrequestpytest 接口自动化测试框架入门 - 简书 pytest和unittest的比较: pytest是一个非常成熟的全功能的Python测试框架,主要有以下几个特点: 简单灵活,容易上手支持参数化能够支持简单的单元测试和复杂的功能测试&a…...
数据结构——复杂度
总有一天你要一个人,再暗夜中,向那座桥走过去 文章目录 一、算法的复杂度 考察形式范例 二、算法的时间复杂度 大O的渐进表示法 常见的复杂度对比 例题:消失的数字 题目的三种思路 1.排序遍历 2.减法 3.单身狗思想 三、空间复杂度…...
使用goldengate 迁移Oracle到postgresql
环境: --源端: IP:10.0.4.16 hostname:tencent Oracle数据库版本:12.2.0.1.0 ogg for oracle版本:19.1.0.0.4 SID:orcl --目标端: IP:10.0.4.16 hostname&#…...
ESP-C3入门20. CentOS开发环境及Jenkins流水线
一、准备环境 CentOS8已经正常安装Jenkins 二、升级 cmake cmake 升到 3.16以上。 cmake --version # 安装 g sudo yum install gcc-c export CXXg# 安装 CMake 的依赖项 sudo yum install -y openssl-devel# 下载 CMake 源码并进行编译安装 wget https://github.com/Kitwa…...
服务器被爬虫恶意攻击怎么办?
在有预算的情况可以采购第三方服务防火墙,没钱就使用开源的WAF进行防护。 # WAF防火墙的基本防护原理 WAF(Web 应用防火墙)可以使用多种技术来防止恶意爬虫攻击,例如: 1. 黑名单:WAF 可以使用黑名单技术来…...
JavaScript正则表达式之座机号/手机号验证校验规则
引用:https://www.bilibili.com/read/cv18300539/ 本文对利用正则表达式对手机号码进行了验证 支持格式: 座机 :xxx-xxxxxxxx、xxxxxxxxxxxx …座机区号的横杠可有可无 手机:xxxxxxxxxxx JavaScript: var: checkPhone (rule,…...
黑客学习手册(自学网络安全)
一、首先,什么是黑客? 黑客泛指IT技术主攻渗透窃取攻击技术的电脑高手,现阶段黑客所需要掌握的远远不止这些。 二、为什么要学习黑客技术? 其实,网络信息空间安全已经成为海陆空之外的第四大战场,除了国…...
获取非叶子节点的grad(retain_grad()、hook)【为了解决grad值是None的问题】
在调试过程中, 有时候我们需要对中间变量梯度进行监控, 以确保网络的有效性, 这个时候我们需要打印出非叶节点的梯度, 为了实现这个目的, 我们可以通过两种手段进行, 分别是: retain_grad()hook 不过我感觉“hook”比“retain_grad()”要麻烦.....,所以我感觉还是…...
JMeter(八):响应断言详解
响应断言 :对服务器的响应进行断言校验 (1)应用范围: main sample and sub sample, main sample only , sub-sample only , jmeter variable 关于应用范围,我们大多数勾选“main sample only” 就足够了,因为我们一个请求,实质上只有一个请求。但是当我们发一个请求时,…...
【网络编程】IO复用的应用一:非阻塞connect
在connect连接中,若socket以非阻塞的方式进行连接,则系统内设置的TCP三次握手超时时间为0,所以它不会等待TCP三次握手完成,直接返回,错误为EINPROGRESS。 所以,我们可以通过判断connect时返回的错误码是…...
Spring注解开发,bean的作用范围及生命周期、Spring注解开发依赖注入
🐌个人主页: 🐌 叶落闲庭 💨我的专栏:💨 c语言 数据结构 javaweb 石可破也,而不可夺坚;丹可磨也,而不可夺赤。 Spring注解开发 一、注解开发定义Bean二、纯注解开发Bean三…...
C#设计模式之---原型模式
原型模式(Prototype Pattern) 原型模式(Prototype Pattern) 是用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。原型模式是一种创建型设计模式。也就是用一个已经创建的实例作为原型,通过…...
STM32入门学习之外部中断
1.STM32的IO口可以作为外部中断输入口。本文通过按键按下作为外部中断的输入,点亮LED灯。在STM32的19个外部中断中,0-15为外部IO口的中断输入口。STM32的引脚分别对应着0-15的外部中断线。比如,外部中断线0对应着GPIOA.0-GPIOG.0,…...
Jenkins 配置maven和jdk
前提:服务器已经安装maven和jdk 一、在Jenkins中添加全局变量 系统管理–>系统配置–>全局属性–>环境变量 添加三个全局变量 JAVA_HOME、MAVEN_HOME、PATH 二、配置maven 系统管理–>全局工具配置–>maven–>新增 新增配置 三、配置JDK 在系统管…...
Leetcode | Binary search | 22. 74. 162. 33. 34. 153.
22. Generate Parentheses 要意识到只要还有左括号,就可以放到path里。只要右括号数量小于左括号,也可以放进去。就是valid的组合。recurse两次 74. Search a 2D Matrix 看成sorted list就好。直接用m*n表示最后一位的index,并且每次只需要 …...
生命在于折腾——面试问题汇总
这里面的问题都是我参加面试时候遇到的问题,大家就这样看吧。 一、个人情况 1、自我介绍 2、为什么离开上一家公司 3、有没有参加过HVV 4、介绍一下上家公司的项目 5、小程序和公众号渗透测试做过么 6、实习工资多少 7、有挖过漏洞么 二、基础知识 1、信息收集的…...
<Java>Map<String,Object>中解析Object类型数据为数组格式
背景: 前端:入参为字符串和数组类型;通过json字符串传给后台, 后台:后台通过工具解析为Map<String,Object>,然后需要解析出Map里面的数组值做操作; 需求: 入参&…...
【Python】 -- 趣味代码 - 小恐龙游戏
文章目录 文章目录 00 小恐龙游戏程序设计框架代码结构和功能游戏流程总结01 小恐龙游戏程序设计02 百度网盘地址00 小恐龙游戏程序设计框架 这段代码是一个基于 Pygame 的简易跑酷游戏的完整实现,玩家控制一个角色(龙)躲避障碍物(仙人掌和乌鸦)。以下是代码的详细介绍:…...
P3 QT项目----记事本(3.8)
3.8 记事本项目总结 项目源码 1.main.cpp #include "widget.h" #include <QApplication> int main(int argc, char *argv[]) {QApplication a(argc, argv);Widget w;w.show();return a.exec(); } 2.widget.cpp #include "widget.h" #include &q…...
Python如何给视频添加音频和字幕
在Python中,给视频添加音频和字幕可以使用电影文件处理库MoviePy和字幕处理库Subtitles。下面将详细介绍如何使用这些库来实现视频的音频和字幕添加,包括必要的代码示例和详细解释。 环境准备 在开始之前,需要安装以下Python库:…...
【C语言练习】080. 使用C语言实现简单的数据库操作
080. 使用C语言实现简单的数据库操作 080. 使用C语言实现简单的数据库操作使用原生APIODBC接口第三方库ORM框架文件模拟1. 安装SQLite2. 示例代码:使用SQLite创建数据库、表和插入数据3. 编译和运行4. 示例运行输出:5. 注意事项6. 总结080. 使用C语言实现简单的数据库操作 在…...
NFT模式:数字资产确权与链游经济系统构建
NFT模式:数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新:构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议:基于LayerZero协议实现以太坊、Solana等公链资产互通,通过零知…...
多种风格导航菜单 HTML 实现(附源码)
下面我将为您展示 6 种不同风格的导航菜单实现,每种都包含完整 HTML、CSS 和 JavaScript 代码。 1. 简约水平导航栏 <!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport&qu…...
ios苹果系统,js 滑动屏幕、锚定无效
现象:window.addEventListener监听touch无效,划不动屏幕,但是代码逻辑都有执行到。 scrollIntoView也无效。 原因:这是因为 iOS 的触摸事件处理机制和 touch-action: none 的设置有关。ios有太多得交互动作,从而会影响…...
Spring是如何解决Bean的循环依赖:三级缓存机制
1、什么是 Bean 的循环依赖 在 Spring框架中,Bean 的循环依赖是指多个 Bean 之间互相持有对方引用,形成闭环依赖关系的现象。 多个 Bean 的依赖关系构成环形链路,例如: 双向依赖:Bean A 依赖 Bean B,同时 Bean B 也依赖 Bean A(A↔B)。链条循环: Bean A → Bean…...
【JavaSE】多线程基础学习笔记
多线程基础 -线程相关概念 程序(Program) 是为完成特定任务、用某种语言编写的一组指令的集合简单的说:就是我们写的代码 进程 进程是指运行中的程序,比如我们使用QQ,就启动了一个进程,操作系统就会为该进程分配内存…...
Python 高效图像帧提取与视频编码:实战指南
Python 高效图像帧提取与视频编码:实战指南 在音视频处理领域,图像帧提取与视频编码是基础但极具挑战性的任务。Python 结合强大的第三方库(如 OpenCV、FFmpeg、PyAV),可以高效处理视频流,实现快速帧提取、压缩编码等关键功能。本文将深入介绍如何优化这些流程,提高处理…...
