数据结构(二)------单链表

制作不易,三连支持一下呗!!!

文章目录

  • 前言
  • 一.什么是链表
  • 二.链表的分类
  • 三.单链表的实现
  • 总结


前言

上一节,我们介绍了顺序表的实现与一些经典算法。

但是顺序表这个数据结构依然有不少缺陷:

1.顺序表指定位置和头部的插入和删除操作的时间复杂度为o(n)。

2.增容需要重新申请新的空间,拷贝旧数据,释放旧空间有不小的损耗。

3.增容是成倍数的扩容,难免还会存在一定的空间浪费。

问题来了:有没有一种数据结构是可以弥补上述顺序表中存在的一些缺点的。答案是肯定的——链表


一.什么是链表

链表的结构就像火车的一节一节的车厢,每节车厢单独用来容纳乘客,但是彼此之间又用链条相互勾连在一起。同时从第一节车厢想要到达第三节车厢必须要经过第二节车厢。

链表依然属于线性表,它在逻辑结构上是连续的,但是物理结构上是不连续的!

链表在逻辑结构上大概是这样的:

 

而这里的每一节“车厢“,我们称之为”节点(结点)“ 。每个节点又是由两个部分组成:一部分用来存储数据,另一部分则是保存下一个节点的地址,以便我们找到下一个节点。链表就是这样通过指针将物理结构上不连续的数据串联在了一起。

二.链表的分类:

链表的分类依据有三点:

1.带头与不带头

2.循环与不循环

3.单向与双向

通过这三种依据的排列组合,链表共可以分为2*2*2=8种。(我们主要学习单链表(不带头单向不循环链表)与双向链表(带头双向不循环链表,下一节介绍))。

下面我们解释一下这三种依据是什么意思:

1.带头与不带头

带头是指带有头节点的链表,所谓头节点是指第一个节点(但是这个节点不保存有效数据,只记录下一个节点的地址,是一个无效的节点),就像放哨的一样,所以也叫“哨兵卫”。

逻辑结构如下:

 

  不带头就是指从第一个节点开始就存储有有效的数据的链表

 

2.循环与不循环 

不循环链表的结尾一定是NULL,以表示链表已结束。

逻辑结构就是这样:

 

而循环链表 则是将不循环链表首尾相连,让它们形成封闭的图形。

逻辑结构如下:

3.单向与双向 

单向链表就是指通过前驱节点可以找到后继节点,但是无法通过后继节点找到前驱节点

逻辑结构如下:

双向链表则是既可以通过前驱节点找到后继节点,也能通过后继节点找回前驱节点

逻辑结构如下:

三.单链表的实现 

我们这一节则是来介绍单链表的实现过程,完成对单链表进行增,删,查,改的接口。

1.单链表节点的定义

根据前面对单链表的介绍,我们知道单链表由两部分组成,一个是保存数据,另一个是保存下一个节点的地址,所以我们可以自然的写出这种结构。

typedef int SLDataType;//重定义方便我们后续存储数据类型发生改变时,一键替换
typedef struct SListNode
{SLDataType data;struct SListNode *next;
}SLTNode;          //重定义方便我们后续书写

2.单链表的打印 

我们接下来先写一个打印单链表的接口,以方便我们后续调试代码!

由于我们还没有申请节点的接口,所以我们首先先手动申请4个节点,并把它们链在一起。

 

画图的话就是这个样子。

如果我们想打印这串链表,通过分析可知,我们可以通过循环遍历这个链表,判断条件是当前节点是否为NULL。每次打印结束将下个节点作为新的节点来打印。 

代码实现如下:

void SListPrint(SLTNode* node1)
{SLTNode* ps = node1;//直接用node1会导致后面找不到头节点while (ps){printf("%d->", ps->data);ps = ps->next;}printf("NULL\n");
}

 

在VS2022上对上述代码的运行结果如下:

打印结果与我们预期结果相符,没有问题。 

3.创建新节点 

我们后续插入数据时都会创建新的节点,所以我们将这个功能单独拿出来实现一个创建新节点的接口。

SLTNode* SLTBuyNode(SLDataType x)//x是要插入的数据
{SLTNode* nownode = (SLTNode*)malloc(sizeof(SLTNode));if (newnode == NULL){perror("malloc");exit(1);}nownode->data = x;nownode->next = NULL;return nownode;
}

 

4.尾插和头插 

<1>.尾插

 

 当链表不为空时:

假设我们现在要在当前链表后尾插一个节点5。思路就是先创建一个新节点5,通过循环找到当前的尾节点4,将node4->next改为node5。

当链表为空时:

直接将新节点作为头节点即可。

代码实现如下:

void SLTPushBack(SLTNode** pphead, SLDataType x)//注意:这里一定要传址调用,否则链表为空的情况就无法插入成功
{assert(pphead);//pphead不能为空,因为我们在函数中需要对pphead解引用SLTNode* newnode = SLTBuyNode(x);if (*pphead == NULL)//链表为空,将新节点直接作为头节点{*pphead = newnode;return;}SLTNode* ptail = *pphead;while (ptail->next){ptail = ptail->next;}//循环结束后,ptail就是尾节点ptail->next = newnode;
}

测试样例:

 

注意: 如果这里我们直接将头节点的地址传进来,而不是传入指向头节点的指针的地址,那么当头节点为空时,形参pphead是头节点地址的临时拷贝,将pphead赋值为nownode并不会改变实参plist,就会导致永远无法插入第一个节点。

 

<2>.头插 

还是要将5插入到链表中,思路就是先创建一个新节点newnode,让newnode->next=*pphead,并将头节点*pphead赋值为newnode(也就是新的头节点)。

代码实现如下:

void SLTPushFront(SLTNode**pphead, SLDataType x)
{assert(pphead); //pphead不能为空,因为我们在函数中需要对pphead解引用SLTNode* newnode= SLTBuyNode(x);newnode->next = *pphead;*pphead = newnode;
}

测试样例:

 

 

5.头删和尾删 

<1>.头删

假设我们要头删掉1这个节点,整体思路就是重新创建一个变量用来保存头节点的后继节点(因为我们接下来要释放掉头节点,不保存一份会导致找不到第二个节点),然后释放掉原来的头节点,并将其置为NULL。 然后将头节点赋值为原来的第二个节点。

代码实现如下:

void SLTPopFront(SLTNode** pphead)
{assert(pphead);assert(*pphead);//链表不能为空//让第二个节点成为新的头节点,并释放掉旧的头节点SLTNode* next = (*pphead)->next;free(*pphead);*pphead = next;
}

测试样例如下:

 

 

<2>.尾删

假设我们想要尾删掉4这个节点,我们都思路就是通过while循环找到尾节点和尾节点的前驱节点。将尾节点的前驱节点置为NULL,释放掉尾节点,并将指向尾节点的指针置为空。

注意:这样有一个特殊的情况就是当链表中只有一个节点时,头节点也就同时是尾节点,这样是没有尾节点的前驱节点的。所以这种情况需要单独考虑!!!

代码实现如下:

void SLTPopBack(SLTNode** pphead, SLDataType x)
{assert(pphead);assert(*pphead);//链表不能为空SLTNode* prev = NULL;SLTNode* pcur = *pphead;if (pcur->next == NULL)//只有一个节点的情况{free(*pphead);*pphead = NULL;return;}while (pcur->next){prev = pcur;pcur = pcur->next;}prev->next = NULL;//销毁尾节点free(pcur);pcur = NULL;
}

样例测试如下:

 

 

6.查找

思路很简单:遍历链表,遇到要找的值就返回这个节点的地址,找不到返回NULL。

代码实现如下:

SLTNode* SLTFind(SLTNode** pphead, SLDataType x)//查找数据x,并返回节点的地址
{assert(pphead);SLTNode* pcur = *pphead;//遍历链表while (pcur){if (pcur->data == x){return pcur;}pcur = pcur->next;}//找不到返回NULLreturn NULL;
}

 

测试样例 :

7.指定位置的插入和删除 

<1>.指定位置之前插入数据

思路:找到目标节点和目标节点的前驱节点,改变目标节点的前驱节点的方向,将新节点串联进来。

pos可以通过查找接口来获取!!!

代码实现如下:

void SLTInsertFront(SLTNode** pphead, SLTNode* pos, SLDataType x)
{assert(pphead);assert(pos);//指定位置肯定不能为空assert(*pphead);//如果链表为空,指定位置一定为空,所以链表不能为空SLTNode* newnode = SLTBuyNode(x);SLTNode* prev = *pphead;if (pos == *pphead)//特殊情况,特殊处理{SLTPushFront(pphead, x);return;}while (prev->next != pos){prev = prev->next;}prev->next = newnode;newnode->next = pos;
}

这里要注意一种特殊的情况:pos是头节点时,while循环将永远无法找到pos。这里我们直接调用之前实现的头插接口即可! ! ! 

 <2>.指定位置之后插入数据

思路就是:先创建新节点newnode,再将newnode的后继节点赋为指定位置pos的后继节点,再将指定位置pos的后继节点置为newnode。

代码实现如下:

void SLTInsertAfter(SLTNode* pos, SLDataType x)
{assert(pos);SLTNode* newnode = SLTBuyNode(x);newnode->next = pos->next;pos->next = newnode;
}

注意: 我们不需要传入头节点,因为只要有pos就可以找到pos之后的所有节点!!!

<3>.删除指定位置的节点

代码实现如下:

void SLTErase(SLTNode** pphead, SLTNode* pos)
{assert(pphead);assert(*pphead);//链表为空就无法删除,不能为空assert(pos);SLTNode* prev = *pphead;//if (pos == *pphead)//如果pos为头节点,没有前驱节点{//头删SLTPopFront(pphead);return;}while (prev->next != pos){prev = prev->next;}prev->next = pos->next;free(pos);pos = NULL;
}

 

<4>.指定位置之后删除数据

思路和指定位置之后插入数据类似,也是先找到pos和pos->next,再改变链表方向,最后释放pos->next节点。

代码实现如下:

void SLTEraseAfter(SLTNode* pos)
{assert(pos);assert(pos->next);SLTNode* del = pos->next;pos->next = pos->next->next;free(del);del = NULL;
}

 

8.销毁链表 

void SLTDesTroy(SLTNode** pphead)
{assert(pphead);assert(*pphead);SLTNode* pcur = *pphead;while (pcur){SLTNode* next = pcur->next;free(pcur);pcur = next;}*pphead = NULL;
}


总结

单链表的所有代码放在这里,以供参考!!!

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
typedef int SLDataType;//重定义方便我们后续存储数据类型发生改变时,一键替换
typedef struct SListNode
{SLDataType data;struct SListNode *next;
}SLTNode;          //重定义方便我们后续书写void SListPrint(SLTNode* node1)
{SLTNode* ps = node1;while (ps){printf("%d->", ps->data);ps = ps->next;}printf("NULL\n");
}
SLTNode* SLTBuyNode(SLDataType x)//x是要插入的数据
{SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));if (newnode == NULL){perror("malloc");exit(1);}newnode->data = x;newnode->next = NULL;return newnode;
}
void SLTPushBack(SLTNode** pphead, SLDataType x)//注意:这里一定要传址调用,否则链表为空的情况就无法插入成功
{assert(pphead);//pphead不能为空,因为我们在函数中需要对pphead解引用SLTNode* newnode = SLTBuyNode(x);if (*pphead == NULL)//链表为空,将新节点直接作为头节点{*pphead = newnode;return;}SLTNode* ptail = *pphead;while (ptail->next){ptail = ptail->next;}//循环结束后,ptail就是尾节点ptail->next = newnode;
}
void SLTPushFront(SLTNode**pphead, SLDataType x)
{assert(pphead); //pphead不能为空,因为我们在函数中需要对pphead解引用SLTNode* newnode= SLTBuyNode(x);newnode->next = *pphead;*pphead = newnode;
}
void SLTPopBack(SLTNode** pphead)
{assert(pphead);assert(*pphead);//链表不能为空SLTNode* prev = NULL;SLTNode* pcur = *pphead;if (pcur->next == NULL)//只有一个节点的情况{free(*pphead);*pphead = NULL;return;}while (pcur->next){prev = pcur;pcur = pcur->next;}prev->next = NULL;//销毁尾节点free(pcur);pcur = NULL;
}
void SLTPopFront(SLTNode** pphead)
{assert(pphead);assert(*pphead);//链表不能为空//让第二个节点成为新的头节点,并释放掉旧的头节点SLTNode* next = (*pphead)->next;free(*pphead);*pphead = next;
}
SLTNode* SLTFind(SLTNode** pphead, SLDataType x)//查找数据x,并返回节点的地址
{assert(pphead);SLTNode* pcur = *pphead;//遍历链表while (pcur){if (pcur->data == x){return pcur;}pcur = pcur->next;}//找不到返回NULLreturn NULL;
}
void SLTInsertFront(SLTNode** pphead, SLTNode* pos, SLDataType x)
{assert(pphead);assert(pos);//指定位置肯定不能为空assert(*pphead);//如果链表为空,指定位置一定为空,所以链表不能为空SLTNode* newnode = SLTBuyNode(x);SLTNode* prev = *pphead;if (pos == *pphead)//特殊情况,特殊处理{SLTPushFront(pphead, x);return;}while (prev->next != pos){prev = prev->next;}prev->next = newnode;newnode->next = pos;
}
void SLTInsertAfter(SLTNode* pos, SLDataType x)
{assert(pos);SLTNode* newnode = SLTBuyNode(x);newnode->next = pos->next;pos->next = newnode;
}
void SLTErase(SLTNode** pphead, SLTNode* pos)
{assert(pphead);assert(*pphead);//链表为空就无法删除,不能为空assert(pos);SLTNode* prev = *pphead;//if (pos == *pphead)//如果pos为头节点,没有前驱节点{//头删SLTPopFront(pphead);return;}while (prev->next != pos){prev = prev->next;}prev->next = pos->next;free(pos);pos = NULL;
}
void SLTEraseAfter(SLTNode* pos)
{assert(pos);assert(pos->next);SLTNode* del = pos->next;pos->next = pos->next->next;free(del);del = NULL;
}
void SLTDesTroy(SLTNode** pphead)
{assert(pphead);assert(*pphead);SLTNode* pcur = *pphead;while (pcur){SLTNode* next = pcur->next;free(pcur);pcur = next;}*pphead = NULL;
}
int main()
{SLTNode* plist = NULL;SLTPushBack(&plist, 1);SLTPushBack(&plist, 2);SLTPushBack(&plist, 3);SLTPushBack(&plist, 4);SListPrint(plist);SLTNode* Findret = SLTFind(&plist, 2);if (Findret){printf("找到了\n");}else{printf("找不到\n");}return 0;
}

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

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

相关文章

导航页配置服务Dashy本地部署并实现公网远程访问

文章目录 简介1. 安装Dashy2. 安装cpolar3.配置公网访问地址4. 固定域名访问 简介 Dashy 是一个开源的自托管的导航页配置服务&#xff0c;具有易于使用的可视化编辑器、状态检查、小工具和主题等功能。你可以将自己常用的一些网站聚合起来放在一起&#xff0c;形成自己的导航…

基于springboot宠物领养系统

摘要 随着社会的不断发展和人们生活水平的提高&#xff0c;宠物在家庭中的地位逐渐上升&#xff0c;宠物领养成为一种流行的社会现象。为了更好地管理和促进宠物领养的过程&#xff0c;本文基于Spring Boot框架设计和实现了一套宠物领养系统。该系统以用户友好的界面为特点&…

时序分析中的去趋势化方法

时序分析中的去趋势化方法 时序分析是研究随时间变化的数据模式的一门学科。在时序数据中&#xff0c;趋势是一种随着时间推移而呈现的长期变化趋势&#xff0c;去趋势化是为了消除或减弱这种趋势&#xff0c;使数据更具平稳性。本文将简单介绍时序分析中常用的去趋势化方法&a…

跟着cherno手搓游戏引擎【13】着色器(shader)

创建着色器类&#xff1a; shader.h:初始化、绑定和解绑方法&#xff1a; #pragma once #include <string> namespace YOTO {class Shader {public:Shader(const std::string& vertexSrc, const std::string& fragmentSrc);~Shader();void Bind()const;void Un…

怎样自行搭建幻兽帕鲁游戏联机服务器?

幻兽帕鲁是一款深受玩家喜爱的多人在线游戏&#xff0c;为了获取更好的游戏体验&#xff0c;许多玩家希望能够自行搭建幻兽帕鲁游戏联机服务器&#xff0c;本文将指导大家如何自行搭建幻兽帕鲁游戏联机服务器。 自行搭建幻兽帕鲁游戏联机服务器&#xff0c;阿里云是一个不错的选…

结构体的增删查改

结构体&#xff0c;是为了解决生活中的一些不方便利用c语言自带数据类型来表示的问题。例如表示一个学生&#xff0c;那么学生这个个体假如用c语言自带数据类型怎么表示呢。可以使用名字&#xff0c;也就是字符数组&#xff1b;也可以使用学号&#xff0c;也就是int类型。但是这…

iOS 面试 Swift基础题

一、Swift 存储属性和计算属性比较&#xff1a; 存储型属性:用于存储一个常量或者变量 计算型属性: 计算性属性不直接存储值,而是用 get / set 来取值 和 赋值,可以操作其他属性的变化. 计算属性可以用于类、结构体和枚举&#xff0c;存储属性只能用于类和结构体。存储属性可…

检测头篇 | 原创自研 | YOLOv8 更换 SEResNeXtBottleneck 头 | 附详细结构图

左图:ResNet 的一个模块。右图:复杂度大致相同的 ResNeXt 模块,基数(cardinality)为32。图中的一层表示为(输入通道数,滤波器大小,输出通道数)。 1. 思路 ResNeXt是微软研究院在2017年发表的成果。它的设计灵感来自于经典的ResNet模型,但ResNeXt有个特别之处:它采用…

MySQL-窗口函数 简单易懂

窗口函数 考查知识点&#xff1a; • 如何用窗口函数解决排名问题、Top N问题、前百分之N问题、累计问题、每组内比较问题、连续问题。 什么是窗口函数 窗口函数也叫作OLAP&#xff08;Online Analytical Processing&#xff0c;联机分析处理&#xff09;函数&#xff0c;可…

Android 基础技术——列表卡顿问题如何分析解决

笔者希望做一个系列&#xff0c;整理 Android 基础技术&#xff0c;本章是关于列表卡顿问题如何分析解决 onBindViewHolder 优化 是否有耗时操作、重复创建对象、设置监听器、findViewByID、局部的动画对象等操作 是否存在内存泄漏 发生内存泄露&#xff0c;会导致一些不再使用…

游戏开发丨基于Tkinter的扫雷小游戏

文章目录 写在前面扫雷小游戏需求分析程序设计程序分析运行结果系列文章写在后面 写在前面 本期内容 基于tkinter的扫雷小游戏 所需环境 pythonpycharm或anaconda 下载地址 https://download.csdn.net/download/m0_68111267/88790713 扫雷小游戏 扫雷是一款广为人知的单…

RabbitMQ“延时队列“

1.RabbitMQ"延时队列" 延迟队列存储的对象是对应的延迟消息&#xff0c;所谓“延迟消息”是指当消息被发送以后&#xff0c;并不想让消费者立刻拿到消息&#xff0c;而是等待特定时间后&#xff0c;消费者才能拿到这个消息进行消费 注意RabbitMQ并没有延时队列慨念,…

OpenCV-29 自适应阈值二值化

一、引入 在前面的部分我们使用的是全局阈值&#xff0c;整幅图像采用同一个数作为阈值。当时这种方法并不适应于所有情况。尤其是当同一幅图像上的不同部分具有不同的亮度时。这种情况下我们需要采用自适应阈值。此时的阈值时根据图像上的每一个小区域计算与其对应的阈值。因此…

【幻兽帕鲁】开服务器,高性能高带宽(100mbps),免费!!!【学生党强推】

【幻兽帕鲁】开服务器&#xff0c;高性能高带宽&#xff08;100mbps&#xff09;&#xff0c;免费&#xff01;&#xff01;&#xff01;【学生党强推】 教程相关视频地址&#xff1a;https://www.bilibili.com/video/BV16e411Y7Fd/ 目前幻兽帕鲁开服务器有以下几套比较性价比的…

研发日记,Matlab/Simulink避坑指南(九)——可变数组应用Bug

文章目录 前言 背景介绍 问题描述 分析排查 解决方案 总结归纳 前言 见《研发日记&#xff0c;Matlab/Simulink避坑指南(四)——transpose()转置函数Bug》 见《研发日记&#xff0c;Matlab/Simulink避坑指南(五)——CAN解包 DLC Bug》 见《研发日记&#xff0c;Matlab/Si…

qemu + vscode图形化调试linux kernel

一、背景 使用命令行连接gdb 在调试时&#xff0c;虽然可以通过tui enable 显示源码&#xff0c;但还是存在设置断点麻烦&#xff08;需要对着源码设置&#xff09;&#xff0c;terminal显示代码不方便&#xff0c;不利于我们学习&#xff1b;另外在gdb 下p命令显示结构体内容…

重构改善既有代码的设计-学习(六):处理继承关系

1、函数上移&#xff08;Pull Up Method&#xff09; 无论何时&#xff0c;只要系统内出现重复&#xff0c;你就会面临“修改其中一个却未能修改另一个”的风险。通常&#xff0c;找出重复也有一定的难度。 所以&#xff0c;某个函数在各个子类中的函数体都相同&#xff08;它们…

Pandas--数据结构 - Series(3)

Pandas Series 类似表格中的一个列&#xff08;column&#xff09;&#xff0c;类似于一维数组&#xff0c;可以保存任何数据类型。 Series 特点&#xff1a; 索引&#xff1a; 每个 Series 都有一个索引&#xff0c;它可以是整数、字符串、日期等类型。如果没有显式指定索引&…

Android Automotive:在路上释放 Android 操作系统的力量

Android Automotive&#xff1a;在路上释放 Android 操作系统的力量 Android 在汽车行业的历程车载信息娱乐系统 (IVI) 的演变汽车中的 Android&#xff1a;演变和进步Android 汽车操作系统的崛起Polestar 2&#xff1a;开创 Android 汽车体验Android 开源项目 (AOSP) 及其他项…

华为三层交换机与防火墙对接配置上网示例

三层交换机与防火墙对接上网配置示例 组网图形 图1 三层交换机与防火墙对接上网组网图 三层交换机简介配置注意事项组网需求配置思路操作步骤配置文件 三层交换机简介 三层交换机是具有路由功能的交换机&#xff0c;由于路由属于OSI模型中第三层网络层的功能&#xff0c;所以…