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

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

文章目录

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


前言

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

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

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;
	}
    //找不到返回NULL
	return 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;
	}
	//找不到返回NULL
	return 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;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

(2)(2.9) Holybro Microhard P900无线电遥测设备

文章目录 前言 1 特点 2 规格 3 包装内包括 前言 Holybro Microhard Radio 集成了 microhard Pico 系列射频模块&#xff0c;能够在强大的拓扑结构中提供高性能无线串行通信&#xff0c;如点对点、点对多点和安全 Mesh&#xff08;P840 不提供 Mesh&#xff09;。 它采用跳…

多线程 之 静态代理

什么是静态代理&#xff1f; 静态代理是一种思想&#xff0c;找一个代理负责一些琐事&#xff0c;自己则专注于一件大事。 有哪些具体的表现&#xff1f; 在日常生活中做饭就是这样&#xff0c;会做饭的人需要做饭&#xff0c;那么其他的人就来帮他打杂&#xff0c;这样做饭的…

Sqli-labs-master第一关通关攻略

第一关基于错误的字符串/数字型注入 第一关打开&#xff0c;请输入id数值作为参数&#xff0c;那就输呗整个1&#xff0c;2&#xff0c;3看看效果 通过ID数值得变动&#xff0c;页面也随之发生变化&#xff0c;然后就是判断SQL语句是否拼接&#xff0c;是字符型还是数字型 输入…

DETR解读,将Transformer带入CV

论文出处 [2005.12872] End-to-End Object Detection with Transformers (arxiv.org) 一个前置知识 匈牙利算法&#xff1a;来源于二部图匹配&#xff0c;计算最小或最大匹配 算法操作&#xff1a;在n*n的矩阵中 减去行列最小值&#xff0c;更新矩阵&#xff08;此时行或者…

(蓝桥杯每日一题)求最长回文串

问题描述 给出一个长度为 n 的小写字符串&#xff0c;求一个最长的子串 S&#xff0c;满足SXY,X&#xff0c;Y>1&#xff0c;且X,Y 均为回文串。 输入格式 输入包括一行: 第一行是一个长度为 n 的小写字符串。 输出格式 输出包括一行&#xff1a; 一行一个整数&#xff0c;表…

Java设计模式-享元模式(12)

馆长准备了很多学习资料,其中包含java方面,jvm调优,spring / spring boot /spring cloud ,微服务,分布式,前端,js书籍资料,视频资料,以及各类常用软件工具,破解工具 等资源。请关注“IT技术馆”公众号,进行关注,馆长会每天更新资源和更新技术文章等。请大家多多关注…

Vue ECharts X轴 type为value的数据格式 + X轴固定间隔并向上取整十位数 - 附完整实例

echarts&#xff1a;一个基于 JavaScript 的开源可视化图表库。 目录 效果 一、介绍 1、官方文档&#xff1a;Apache ECharts 2、官方示例 二、准备工作 1、安装依赖包 2、示例版本 三、使用步骤 1、在单页面引入 echarts 2、指定容器并设置容器宽高 3、数据处理&am…

Java: javax.net.ssl.SSLPeerUnverifiedException: peer not authenticated

我们在平时练习的时候一般使用低版本的jdk来练习&#xff0c;以便了解不同版本jdk的区别&#xff0c;下面是我们练习中遇到的问题 >>> DefaultHttpClient mHttpClient new DefaultHttpClient(new BasicHttpParams()); ClientConnectionManager ccm mHttpClien…

【Vite+Vue3+TS】基于Vite+Vue3+TypeScript+ESLint+Prettier+Stylelint搭建项目(亲测超详细)

目 录 项目搭建步骤确定node版本使用Vite创建Vue3项目规范目录结构配置环境修改Vite配置文件集成路由工具Vue Router集成状态管理工具Pinia集成CSS预编译器Sassvite-plugin-svg-icons图标组件集成UI框架Element Plus集成HTTP 请求工具 Axios 项目代码规范集成ESLint配置集成Pre…

docker环境搭建及其安装常用软件

centos安装docker Install Docker Engine on CentOS | Docker Docs 下载docker sudo yum install -y yum-utils sudo yum-config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo sudo yum install -y docker-ce docker-ce-cli containerd.io…

Git学习,基础,安装,配置,笔记总结

Git安装与常用命令 本教程里的git命令例子都是在Git Bash中演示的,会用到一些基本的linux命令,在此为大家提前列举: ls/ll 查看当前目录 cat 查看文件内容 touch 创建文件 vi vi编辑器(使用vi编辑器是为了方便展示效果,学员可以记事本、editPlus、notPad++等其它编 辑…

【YOLO系列算法俯视视角下舰船目标检测】

YOLO系列算法俯视视角下舰船目标检测 数据集和模型YOLO系列算法俯视视角下舰船目标检测YOLO系列算法俯视视角下舰船目标检测可视化结果 数据集和模型 数据和模型下载&#xff1a; YOLOv6俯视视角下舰船目标检测训练好的舰船目标检测模型舰船目标检测数据YOLOv7俯视视角下舰船…

ES6.8.6 为索引映射(Mapping)创建自定义分词器,测试分词匹配效果

文章目录 环境创建索引&#xff1a;配置自定义分词器、字段指定分词器自定义分词器参数说明创建索引&#xff1a;custom_analyzer_comment 使用索引中自定义的分词器进行分词分析自定义分词器my_custom_analyzer分词测试&#xff1a;测试中文停用词、英文字母转小写测试敏感词替…

Parquet文件推送数据到OSS

1. 任务背景 任务说明&#xff1a;公司 saas 数据分析类产品&#xff0c;客户需要把行为数据回传到客户指定文件系统中&#xff08;oss&#xff09;周期&#xff1a;T1数据格式&#xff1a;parquet数据范围&#xff1a;部分表全量&#xff0c;部分表增量其他要求&#xff1a; …

STM32-LwESP 移植

LwESP 是一个专门解析 Espressif 公司旗下 ESP 系列芯片 AT 指令的开源库&#xff0c;具有以下特性&#xff1a; 支持 Espressif 公司 ESP32, ESP32-C2, ESP32-C3, ESP32-C6 和 ESP8266 芯片。独立平台&#xff0c;采用 C99 标准编写&#xff0c;易于移植。允许不同的配置来优…

【Linux】第三十九站:可重入函数、volatile、SIGCHLD信号

文章目录 一、可重入函数二、volatile三、SIGCHLD信号 一、可重入函数 如下图所示&#xff0c;当我们进行链表的头插的时候&#xff0c;我们刚刚执行完第一条语句的时候&#xff0c;突然收到一个信号&#xff0c;然后我们这个信号的自定义捕捉方法中&#xff0c;正好还有一个头…

Python模拟艾里光束:光可以不沿直线传播

文章目录 Airy光束有限能量Airy光束 Airy光束 在光学领域&#xff0c;傍轴近似下光束传输遵循方程 i ∂ ϕ ∂ z 1 z a ∂ 2 ϕ ∂ x 2 0 i\frac{\partial\phi}{\partial z}\frac{1}{z}\frac{a\partial^2\phi}{\partial x^2}0 i∂z∂ϕ​z1​∂x2a∂2ϕ​0 其中 k 2 π n …

【发展】不确定时代下的从容 —— 终局思维、长期主义与复利

文章目录 一、终局思维1、电影 《蝴蝶效应》2、未来是什么样的 二、长期主义1、这是一个不确定的时代2、做难但正确的事情 三、复利1、复利思维2、马太效应 一、终局思维 终局思维 在面对很多选择时&#xff0c;从终点出发考虑问题&#xff0c;来决定当下的选择。 1、电影 《蝴…

容器和虚拟机的对比

容器和虚拟机的对比 容器和虚拟机在与硬件和底层操作系统交互的方式上有所不同 虚拟化 使多个操作系统能够同时在一个硬件平台上运行。 使用虚拟机监控程序将硬件分为多个虚拟硬件系统&#xff0c;从而允许多个操作系统并行运行。 需要一个完整的操作系统环境来支持该应用。…

从零开始:CentOS系统下搭建DNS服务器的详细教程

前言 如果你希望在CentOS系统上建立自己的DNS服务器,那么这篇文章绝对是你不容错过的宝藏指南。我们提供了详尽的步骤和实用技巧,让你能够轻松完成搭建过程。从安装必要的软件到配置区域文件,我们都将一一为你呈现。无论你的身份是运维人员,还是程序员,抑或是对网络基础设…