当前位置: 首页 > news >正文

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

在这里插入图片描述

💐 🌸 🌷 🍀 🌹 🌻 🌺 🍁 🍃 🍂 🌿 🍄🍝 🍛 🍤
📃个人主页 :阿然成长日记 👈点击可跳转
📆 个人专栏: 🔹数据结构与算法🔹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)&#xf…...

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类型数据为数组格式

背景&#xff1a; 前端&#xff1a;入参为字符串和数组类型&#xff1b;通过json字符串传给后台&#xff0c; 后台&#xff1a;后台通过工具解析为Map<String&#xff0c;Object>&#xff0c;然后需要解析出Map里面的数组值做操作&#xff1b; 需求&#xff1a; 入参&…...

web vue 项目 Docker化部署

Web 项目 Docker 化部署详细教程 目录 Web 项目 Docker 化部署概述Dockerfile 详解 构建阶段生产阶段 构建和运行 Docker 镜像 1. Web 项目 Docker 化部署概述 Docker 化部署的主要步骤分为以下几个阶段&#xff1a; 构建阶段&#xff08;Build Stage&#xff09;&#xff1a…...

盘古信息PCB行业解决方案:以全域场景重构,激活智造新未来

一、破局&#xff1a;PCB行业的时代之问 在数字经济蓬勃发展的浪潮中&#xff0c;PCB&#xff08;印制电路板&#xff09;作为 “电子产品之母”&#xff0c;其重要性愈发凸显。随着 5G、人工智能等新兴技术的加速渗透&#xff0c;PCB行业面临着前所未有的挑战与机遇。产品迭代…...

Keil 中设置 STM32 Flash 和 RAM 地址详解

文章目录 Keil 中设置 STM32 Flash 和 RAM 地址详解一、Flash 和 RAM 配置界面(Target 选项卡)1. IROM1(用于配置 Flash)2. IRAM1(用于配置 RAM)二、链接器设置界面(Linker 选项卡)1. 勾选“Use Memory Layout from Target Dialog”2. 查看链接器参数(如果没有勾选上面…...

【开发技术】.Net使用FFmpeg视频特定帧上绘制内容

目录 一、目的 二、解决方案 2.1 什么是FFmpeg 2.2 FFmpeg主要功能 2.3 使用Xabe.FFmpeg调用FFmpeg功能 2.4 使用 FFmpeg 的 drawbox 滤镜来绘制 ROI 三、总结 一、目的 当前市场上有很多目标检测智能识别的相关算法&#xff0c;当前调用一个医疗行业的AI识别算法后返回…...

鸿蒙DevEco Studio HarmonyOS 5跑酷小游戏实现指南

1. 项目概述 本跑酷小游戏基于鸿蒙HarmonyOS 5开发&#xff0c;使用DevEco Studio作为开发工具&#xff0c;采用Java语言实现&#xff0c;包含角色控制、障碍物生成和分数计算系统。 2. 项目结构 /src/main/java/com/example/runner/├── MainAbilitySlice.java // 主界…...

【7色560页】职场可视化逻辑图高级数据分析PPT模版

7种色调职场工作汇报PPT&#xff0c;橙蓝、黑红、红蓝、蓝橙灰、浅蓝、浅绿、深蓝七种色调模版 【7色560页】职场可视化逻辑图高级数据分析PPT模版&#xff1a;职场可视化逻辑图分析PPT模版https://pan.quark.cn/s/78aeabbd92d1...

【Redis】笔记|第8节|大厂高并发缓存架构实战与优化

缓存架构 代码结构 代码详情 功能点&#xff1a; 多级缓存&#xff0c;先查本地缓存&#xff0c;再查Redis&#xff0c;最后才查数据库热点数据重建逻辑使用分布式锁&#xff0c;二次查询更新缓存采用读写锁提升性能采用Redis的发布订阅机制通知所有实例更新本地缓存适用读多…...

【 java 虚拟机知识 第一篇 】

目录 1.内存模型 1.1.JVM内存模型的介绍 1.2.堆和栈的区别 1.3.栈的存储细节 1.4.堆的部分 1.5.程序计数器的作用 1.6.方法区的内容 1.7.字符串池 1.8.引用类型 1.9.内存泄漏与内存溢出 1.10.会出现内存溢出的结构 1.内存模型 1.1.JVM内存模型的介绍 内存模型主要分…...

嵌入式学习之系统编程(九)OSI模型、TCP/IP模型、UDP协议网络相关编程(6.3)

目录 一、网络编程--OSI模型 二、网络编程--TCP/IP模型 三、网络接口 四、UDP网络相关编程及主要函数 ​编辑​编辑 UDP的特征 socke函数 bind函数 recvfrom函数&#xff08;接收函数&#xff09; sendto函数&#xff08;发送函数&#xff09; 五、网络编程之 UDP 用…...

macOS 终端智能代理检测

&#x1f9e0; 终端智能代理检测&#xff1a;自动判断是否需要设置代理访问 GitHub 在开发中&#xff0c;使用 GitHub 是非常常见的需求。但有时候我们会发现某些命令失败、插件无法更新&#xff0c;例如&#xff1a; fatal: unable to access https://github.com/ohmyzsh/oh…...