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

【数据结构】单链表(一)

上一篇【数据结构】顺序表-CSDN博客  我们了解了顺序表,但是呢顺序表涉及到了一些问题,比如,中间/头部的插入/删除,时间复杂度为O(N);增容申请空间、拷贝、释放旧空间会有不小的消耗;增容所浪费的空间... 我们如何去解决这些问题?本篇介绍另一个数据结构——链表

1. 链表的概念及结构

链表:是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针连接次序实现的,链表也是线性表的一种

链表大概是什么样的呢?看下图

链表是由一个一个的节点组成,再顺序表中,数据是连续存放的,我们想找到下一个数据顺着前一个数据找就行了,而链表中数据存放空间是不连续的,所以我们就需要一个指针来保存下一个节点的地址,所以,我们如果要定义链表,只需要定义链表的节点结构就行了

(在【C语言】结构体详解-CSDN博客 的1.3 结构体的自引用 中介绍过)

struct SListNode
{int data;//数据struct SListNode* next;//下一个数据的地址
};

和上一篇一样,创建一个头文件,两个源文件

 也是一样,在头文件SList.h中定义单链表的结构,并对类型和结构体改名

#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
typedef int Type; 
//定义节点结构
typedef struct SListNode
{Type data;struct SListNode* next;
}SLNode;

我们先创建几个节点,在test.c中实现

#include "SList.h" //包含头文件
void SListtest1()
{SLNode* node1 = (SLNode*)malloc(sizeof(SLNode));//创建节点1node1->data = 1;//赋值SLNode* node2 = (SLNode*)malloc(sizeof(SLNode));//创建节点2node2->data = 2;//赋值SLNode* node3 = (SLNode*)malloc(sizeof(SLNode));//创建节点3node3->data = 3;//赋值SLNode* node4 = (SLNode*)malloc(sizeof(SLNode));//创建节点4node4->data = 4;//赋值
}
int main()
{SListtest1();return 0;
}

虽然我们创建好了节点,但是这几个节点现在还不能互相找到,所以我们接下来要将这几个节点连接起来,怎么连接?存下一个节点的地址

void SListtest1()
{SLNode* node1 = (SLNode*)malloc(sizeof(SLNode));//创建节点1node1->data = 1;//赋值SLNode* node2 = (SLNode*)malloc(sizeof(SLNode));//创建节点2node2->data = 2;//赋值SLNode* node3 = (SLNode*)malloc(sizeof(SLNode));//创建节点3node3->data = 3;//赋值SLNode* node4 = (SLNode*)malloc(sizeof(SLNode));//创建节点4node4->data = 4;//赋值//将四个节点连接起来node1->next = node2;node2->next = node3;node3->next = node4;node4->next = NULL;
}

现在就构成了一个链表,有了这个简单的链表,我们来写一个函数打印一下里面的数据

链表的打印

SList.h中进行函数的声明

void SLPrint(SLNode* ps);//打印

参数是链表的节点的地址 

SList.c中进行函数的实现

#include "SList.h"
void SLPrint(SLNode* ps) //打印
{SLNode* pcur = ps;//pcur指向当前链表的第一个节点
}

要打印当然就要循环遍历,写一个while循环,判断条件为pcur,不为NULL进入循环 

void SLPrint(SLNode* ps) //打印
{SLNode* pcur = ps;//pcur指向当前链表的第一个节点while (pcur) //判断第一个节点是否为NULL{//....}
}

 进入循环后打印内容

void SLPrint(SLNode* ps) //打印
{SLNode* pcur = ps;//pcur指向当前链表的第一个节点while (pcur){printf("%d->", pcur->data);//打印内容}
}

打印完一个数据后,要把下一个节点值部分的地址给pcur ,也就是pcur要存node2中2的地址,此时pcur不再指向node1,指向了node2

void SLPrint(SLNode* ps) //打印
{SLNode* pcur = ps;//pcur指向当前链表的第一个节点while (pcur){printf("%d->", pcur->data);//打印内容pcur = pcur->next;//指向下一个节点}printf("NULL\n");
}

test.c中测试

测试时我们不直接传链表的第一个节点node1,而是再定义一个结构体指针plist去指向node1,让plist作为参数传过去

    SLNode* plist = node1;SLPrint(plist);

虽然有点多此一举,但这样会让我们的代码更完善 

运行结果

 2.链表的插入

实际使用链表的时候我们不会像上面那样一个一个创建,初始时候是一个空链表,会跟顺序表类似进行空间开辟然后插入数据

2.1空间申请创建节点

插入数据前依旧是要判断空间够不够

SList.h中进行函数的声明

SLNode* SLBuyNode(Type x);//申请空间创建节点

返回值是节点的地址,参数就是要插入的数据

SList.c中进行函数的实现

SLNode* SLBuyNode(Type x)//申请空间创建节点
{SLNode* newnode = (SLNode*)malloc(sizeof(SLNode));if (newnode == NULL) //空间申请失败{perror("malloc");return 1;}newnode->data = x;newnode->next = NULL;return newnode;
}

2.2 尾插

SList.h中进行函数的声明

void SLPushBack(SLNode** ps, Type x);//尾插

参数是链表的节点地址,是一个二级指针,和要插入的数据

我们要给函数传地址,这样形参的改变才能影响实参,对参数的理解如下,这很重要,这里提前展示将要在test.c中测试用的部分代码 

SList.c中进行函数的实现

尾插要先找到尾节点,再将尾节点和新节点连接起来 ,此时node4的地址部分就不再是NULL 而是新节点的地址

代码如何实现呢 ?先找尾节点

void SLPushBack(SLNode* ps, Type x)//尾插
{SLNode* ptail = ps;//定义尾节点,开始时指向头节点while (ptail->next != NULL)//遍历链表数据,找尾节点{ptail = ptail->next;}//跳出循环后ptail指向尾节点
}

然后我们直接调用创建节点的函数,放在找尾节点前面,最后赋值

void SLPushBack(SLNode* ps, Type x)//尾插
{SLNode* newnode = SLBuyNode(x);//申请空间创建节点SLNode* ptail = ps;//定义尾节点,开始时指向头节点while (ptail->next != NULL)//遍历链表数据,找尾节点{ptail = ptail->next;}//跳出循环后ptail指向尾节点ptail->next = newnode;//直接赋值
}

 但是呢,链表在没有赋值的时候是空链表,所以我们要讨论空链表的情况,如果是空链表,直接让ps指向新节点newnode,所以最终的代码如下

void SLPushBack(SLNode** pps, Type x)//尾插
{assert(pps);//pps不可以为NULLSLNode* newnode = SLBuyNode(x);//申请空间创建节点if (*pps == NULL) //*pps可以为NULL,表示空链表{*pps = newnode;}else //非空链表{SLNode* ptail = *pps;//定义尾节点,开始时指向头节点while (ptail->next)//遍历链表数据,找尾节点{ptail = ptail->next;}//跳出循环后ptail指向尾节点ptail->next = newnode;//直接赋值}
}

我们再来看test.c中测试的代码

void SListtest2()
{SLNode* plist = NULL;//空链表SLPushBack(&plist, 1);//尾插SLPushBack(&plist, 2);SLPushBack(&plist, 3);SLPushBack(&plist, 4);SLPrint(plist);//打印
}
int main()
{//SListtest1();SListtest2();return 0;
}

多插入几个数据,运行看结果,插入成功

2.3 头插

 在SList.h中进行函数的声明

void SLPushHead(SLNode** pps, Type x);//头插

同样的,参数也是一个二级指针,还有要插入的数据

SList.c中进行函数的实现

我们需要做两件事,一个是将newnode和原本的首节点连接在一起,另一件事是*pps要指向新的首节点 

 链表不为NULL时,代码如下

void SLPushHead(SLNode** pps, Type x)//头插
{assert(pps);//pps不能为NULLSLNode* newnode = SLBuyNode(x);//申请空间创建节点newnode->next = *pps;*pps = newnode;
}

当链表为NULL时,分析一下

当前代码也可以实现

test.c中测试

void SListtest2()
{SLNode* plist = NULL;//空链表SLPushBack(&plist, 1);//尾插SLPushBack(&plist, 2);SLPushBack(&plist, 3);SLPushBack(&plist, 4);SLPrint(plist);//打印SLPushHead(&plist, 6);//头插SLPushHead(&plist, 7);SLPushHead(&plist, 8);SLPrint(plist);//打印
}

看一下运行结果

3.链表的删除

删除数据,链表就不能为空,不然删啥呢?

 3.1 尾删

SList.h中进行函数的声明

void SLPopBack(SLNode** pps);//尾删

SList.c中进行函数的实现

很显然,首先就是找尾节点,找到尾节点之后直接释放吗?node4释放后node3->next里面依然存放着node4的地址,但此时指向的空间已经没了,此时的指针就成了一个野指针 ,所以我们还要将被释放节点的前一个节点的指针置空

所以我们还要找尾节点的前一个节点

void SLPopBack(SLNode** pps)//尾删
{assert(pps && *pps);//pps和链表都不能为空SLNode* prev = *pps;//定义尾节点前一个节点SLNode* ptail = *pps;//定义尾节点while (ptail->next){prev = ptail;ptail = ptail->next;}//此时找到了尾节点和尾节点前一个节点free(ptail);//释放尾节点ptail = NULL;//置空prev->next = NULL;//置空尾节点前一个节点
}

如果这个链表只有一个节点呢,这个代码可行吗?来分析一下

我们把节点释放并置空后,prev->next就不存在了,函数最后一句代码就不能实现,所以,当链表里只有一个节点时,直接释放就行了

void SLPopBack(SLNode** pps)//尾删
{assert(pps && *pps);//pps和链表都不能为空if ((*pps)->next == NULL)//链表只有一个节点{free(*pps);//释放节点*pps = NULL;//置空}else //链表不止一个节点{SLNode* prev = *pps;//定义尾节点前一个节点SLNode* ptail = *pps;//定义尾节点while (ptail->next){prev = ptail;ptail = ptail->next;}//此时找到了尾节点和尾节点前一个节点free(ptail);//释放尾节点ptail = NULL;//置空prev->next = NULL;//置空尾节点前一个节点}
}

test.c中测试

void SListtest2()
{SLNode* plist = NULL;//空链表SLPushBack(&plist, 1);//尾插SLPushBack(&plist, 2);SLPrint(plist);//打印SLPushHead(&plist, 6);//头插SLPushHead(&plist, 7);SLPrint(plist);//打印SLPopBack(&plist);//尾删SLPrint(plist);//打印
}

看下结果,尾删成功

3.2 头删

SList.h中进行函数的声明

void SLPopHead(SLNode** pps);//头删

SList.c中进行函数的实现

我们要删除头节点,跟删除尾节点不一样,如果我们把首节点释放掉,还能找到首节点里的next吗?不能,不能的话就找不到第二个节点以及剩下的节点

我们应该先把下一个节点的信息存储起来

SLNode* Next = (*pps)->next;//存节点

 然后把原头节点释放

free(*pps);

最后让*pps指向新的头节点

*pps = Next;

所以代码如下

void SLPopHead(SLNode** pps)//头删
{assert(pps && *pps);//pps和链表都不能为空SLNode* Next = (*pps)->next;//存节点free(*pps);*pps = Next;
}

当链表只有一个节点时,分析一下,上面的代码也是可以完成的

我们在test.c中测试一下

void SListtest2()
{SLNode* plist = NULL;//空链表SLPushBack(&plist, 1);//尾插SLPushBack(&plist, 2);SLPrint(plist);//打印SLPushHead(&plist, 6);//头插SLPushHead(&plist, 7);SLPrint(plist);//打印SLPopBack(&plist);//尾删SLPrint(plist);//打印SLPopHead(&plist);//头删SLPrint(plist);//打印
}

下篇我们再继续介绍更多内容,拜拜~

相关文章:

【数据结构】单链表(一)

上一篇【数据结构】顺序表-CSDN博客 我们了解了顺序表&#xff0c;但是呢顺序表涉及到了一些问题&#xff0c;比如&#xff0c;中间/头部的插入/删除&#xff0c;时间复杂度为O(N);增容申请空间、拷贝、释放旧空间会有不小的消耗&#xff1b;增容所浪费的空间... 我们如何去解…...

SCI一区 | Matlab实现INFO-TCN-BiGRU-Attention向量加权算法优化时间卷积双向门控循环单元注意力机制多变量时间序列预测

SCI一区 | Matlab实现INFO-TCN-BiGRU-Attention向量加权算法优化时间卷积双向门控循环单元注意力机制多变量时间序列预测 目录 SCI一区 | Matlab实现INFO-TCN-BiGRU-Attention向量加权算法优化时间卷积双向门控循环单元注意力机制多变量时间序列预测预测效果基本介绍模型描述程…...

Coursera吴恩达《深度学习》课程总结(全)

这里有Coursera吴恩达《深度学习》课程的完整学习笔记&#xff0c;一共5门课&#xff1a;《神经网络和深度学习》、《改善深层神经网络》、《结构化机器学习项目》、《卷积神经网络》和《序列模型》&#xff0c; 第一门课&#xff1a;神经网络和深度学习基础&#xff0c;介绍一…...

C# 操作PDF表单 - 创建、填写、删除PDF表单域

通常情况下&#xff0c;PDF文件是不可编辑的&#xff0c;但PDF表单提供了一些可编辑区域&#xff0c;允许用户填写和提交信息。PDF表单通常用于收集信息、反馈或进行在线申请&#xff0c;是许多行业中数据收集和交换的重要工具。 PDF表单可以包含各种类型的输入控件&#xff0…...

Astropy:探索宇宙奥秘的Python工具箱

Astropy:探索宇宙奥秘的Python工具箱 什么是Astropy库&#xff1f; Astropy 是一个面向天文学领域的 Python 库&#xff0c;旨在提供常用的天文学数据结构、单位转换、物理常数以及天文学计算方法等功能。它为天文学家和研究人员提供了丰富的工具&#xff0c;用于处理和分析天文…...

java数据结构与算法刷题-----LeetCode684. 冗余连接

java数据结构与算法刷题目录&#xff08;剑指Offer、LeetCode、ACM&#xff09;-----主目录-----持续更新(进不去说明我没写完)&#xff1a;https://blog.csdn.net/grd_java/article/details/123063846 文章目录 并查集 并查集 解题思路&#xff1a;时间复杂度O( n ∗ l o g 2…...

循环神经网络简介

卷积神经网络相当于人类的视觉&#xff0c;但是它并没有记忆能力&#xff0c;所以它只能处理一种特定的视觉任务&#xff0c;没办法根据以前的记忆来处理新的任务。比如&#xff0c;在一场电影中推断下一个时间点的场景&#xff0c;这个时候仅依赖于现在的场景还不够&#xff0…...

计算机网络 子网掩码与划分子网

一、实验要求与内容 1、需拓扑图和两个主机的IP配置截图。 2、设置网络A内的主机IP地址为“192.168.班内学号.2”&#xff0c;子网掩码为“255.255.255.128”&#xff0c;网关为“192.168.班内学号.1”&#xff1b;设置网络B内的主机IP地址为“192.168.班内学号100.2”&#…...

HUD抬头显示器中如何设计LCD的阳光倒灌实验

关键词&#xff1a;阳光倒灌实验、HUD光照温升测试、LCD光照温升测试、太阳光模拟器 HUD&#xff08;Head-Up Display&#xff0c;即抬头显示器&#xff09;是一种将信息直接投影到驾驶员视线中的技术&#xff0c;通常用于飞机、汽车等驾驶舱内。HUD系统中的LCD&#xff08;Liq…...

Shoplazza闪耀Shoptalk 2024,新零售创新解决方案引领行业新篇章!

在近期举办的全球零售业瞩目盛事——Shoptalk 2024大会上,全球*的零售技术平台-店匠科技(Shoplazza)以其*的创新实力与前瞻的技术理念,成功吸引了与会者的广泛关注。此次盛会于3月17日至20日在拉斯维加斯曼德勒湾隆重举行,汇聚了逾万名行业精英。在这场零售业的盛大聚会上,Shop…...

Linux:sprintf、snprintf、vsprintf、asprintf、vasprintf比较

这些函数都在stdio.h里&#xff0c;不过不同系统不同库&#xff0c;有些函数不一定提供。 1. sprintf 函数原型&#xff1a; int sprintf (char *str, const char *format, ...); extern int sprintf (char *__restrict __s, const char *__restrict __format, ...); 功能是将…...

Github远程仓库改名字之后,本地git如何配置?

文章目录 缘由解决方案 缘由 今天在github创建一个仓库&#xff0c;备份一下本地电脑上的资料。起初随便起一个仓库名字&#xff0c;后来修改之。既然远程仓库改名&#xff0c;那么本地仓库需要更新地址。这里采用SSH格式。 解决方案 如果你的GitHub仓库改名了&#xff0c;你…...

Objective-C学习笔记(ARC,分类,延展)4.10

1.自动释放池autoreleasepool&#xff1a;存入到自动释放池的对象&#xff0c;在自动释放池销毁时&#xff0c;会自动调用池内所有对象的release方法。调用autorelease方法将对象放入自动释放池。 Person *p1 [ [ [ Person alloc ] init ] autorelease]; 2.在类方法里写一个…...

02 Git 之IDEA 集成使用 GitHub(Git同时管理本地仓库和远程仓库)

2 .IDEA 集成使用 GitHub&#xff08;Git同时管理本地仓库和远程仓库&#xff09; 首先在 IDEA 的设置中绑定 GitHub 的账号 先创建一个 test1.txt 文件&#xff0c;内容为 aaa. 最上一栏 VCS&#xff0c; SHARE ON GitHub&#xff0c;然后选择要发送到远程仓库的文件即可。…...

CSS滚动条样式修改

前言 目前我们可以通过 CSS伪类 来实现滚动条的样式修改&#xff0c;以下为修改滚动条样式用到的CSS伪类&#xff1a; ::-webkit-scrollbar — 整个滚动条 ::-webkit-scrollbar-button — 滚动条上的按钮 (上下箭头) ::-webkit-scrollbar-thumb — 滚动条上的滚动滑块 ::-web…...

《零秒思考》像麦肯锡精英一样思考 - 三余书屋 3ysw.net

零秒思考&#xff1a;像麦肯锡精英一样思考 大家好&#xff0c;今天我们要深入探讨的著作是《零秒思考》。在领导提出问题时&#xff0c;我们常常会陷入沉思&#xff0c;却依然难以有所进展&#xff0c;仿佛原地踏步&#xff0c;但是身边的同事却能够立即给出清晰的回答。这种…...

使用docker制作Android镜像(实操可用)

一、安装包准备 1、准备jdk 下载地址&#xff1a;Java Downloads | Oracle 注意版本&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 我下载的jdk17&#xff0c;不然后面构建镜像报错&#xff0c;就是版本不对 2、准备安装的工具包 ttps://dev…...

大厂MVP技术JAVA架构师培养

课程介绍 这是一个很强悍的架构师涨薪计划课程&#xff0c;课程由专家级MVP讲师进行教学&#xff0c;分为是一个章节进行分解式面试及讲解&#xff0c;不仅仅是面试&#xff0c;更像是一个专业的架构师研讨会课程。课程内容从数据结构与算法、Spring Framwork、JVM原理、 JUC并…...

uniapp实现文件和图片选择上传功能实现

主要介绍了uni-file-picker文件选择上传,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下 上传一张: <template><view class="container example"><uni-forms ref="baseForm" …...

2024认证杯数学建模C题思路模型代码

目录 2024认证杯数学建模C题思路模型代码&#xff1a;4.11开赛后第一时间更新&#xff0c;获取见文末名片 以下为2023年认证杯C题&#xff1a; 2024年认证杯数学建模C题思路模型代码见此 2024认证杯数学建模C题思路模型代码&#xff1a;4.11开赛后第一时间更新&#xff0c;获…...

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…...

springboot 百货中心供应链管理系统小程序

一、前言 随着我国经济迅速发展&#xff0c;人们对手机的需求越来越大&#xff0c;各种手机软件也都在被广泛应用&#xff0c;但是对于手机进行数据信息管理&#xff0c;对于手机的各种软件也是备受用户的喜爱&#xff0c;百货中心供应链管理系统被用户普遍使用&#xff0c;为方…...

调用支付宝接口响应40004 SYSTEM_ERROR问题排查

在对接支付宝API的时候&#xff0c;遇到了一些问题&#xff0c;记录一下排查过程。 Body:{"datadigital_fincloud_generalsaas_face_certify_initialize_response":{"msg":"Business Failed","code":"40004","sub_msg…...

可靠性+灵活性:电力载波技术在楼宇自控中的核心价值

可靠性灵活性&#xff1a;电力载波技术在楼宇自控中的核心价值 在智能楼宇的自动化控制中&#xff0c;电力载波技术&#xff08;PLC&#xff09;凭借其独特的优势&#xff0c;正成为构建高效、稳定、灵活系统的核心解决方案。它利用现有电力线路传输数据&#xff0c;无需额外布…...

【网络安全产品大调研系列】2. 体验漏洞扫描

前言 2023 年漏洞扫描服务市场规模预计为 3.06&#xff08;十亿美元&#xff09;。漏洞扫描服务市场行业预计将从 2024 年的 3.48&#xff08;十亿美元&#xff09;增长到 2032 年的 9.54&#xff08;十亿美元&#xff09;。预测期内漏洞扫描服务市场 CAGR&#xff08;增长率&…...

HTML 列表、表格、表单

1 列表标签 作用&#xff1a;布局内容排列整齐的区域 列表分类&#xff1a;无序列表、有序列表、定义列表。 例如&#xff1a; 1.1 无序列表 标签&#xff1a;ul 嵌套 li&#xff0c;ul是无序列表&#xff0c;li是列表条目。 注意事项&#xff1a; ul 标签里面只能包裹 li…...

定时器任务——若依源码分析

分析util包下面的工具类schedule utils&#xff1a; ScheduleUtils 是若依中用于与 Quartz 框架交互的工具类&#xff0c;封装了定时任务的 创建、更新、暂停、删除等核心逻辑。 createScheduleJob createScheduleJob 用于将任务注册到 Quartz&#xff0c;先构建任务的 JobD…...

土地利用/土地覆盖遥感解译与基于CLUE模型未来变化情景预测;从基础到高级,涵盖ArcGIS数据处理、ENVI遥感解译与CLUE模型情景模拟等

&#x1f50d; 土地利用/土地覆盖数据是生态、环境和气象等诸多领域模型的关键输入参数。通过遥感影像解译技术&#xff0c;可以精准获取历史或当前任何一个区域的土地利用/土地覆盖情况。这些数据不仅能够用于评估区域生态环境的变化趋势&#xff0c;还能有效评价重大生态工程…...

视觉slam十四讲实践部分记录——ch2、ch3

ch2 一、使用g++编译.cpp为可执行文件并运行(P30) g++ helloSLAM.cpp ./a.out运行 二、使用cmake编译 mkdir build cd build cmake .. makeCMakeCache.txt 文件仍然指向旧的目录。这表明在源代码目录中可能还存在旧的 CMakeCache.txt 文件,或者在构建过程中仍然引用了旧的路…...

android13 app的触摸问题定位分析流程

一、知识点 一般来说,触摸问题都是app层面出问题,我们可以在ViewRootImpl.java添加log的方式定位;如果是touchableRegion的计算问题,就会相对比较麻烦了,需要通过adb shell dumpsys input > input.log指令,且通过打印堆栈的方式,逐步定位问题,并找到修改方案。 问题…...