数据结构基础:一篇文章教你单链表(头插,尾插,查找,头删等的解析和代码)

和我一起学编程呀,大家一起努力!

这篇文章耗时比较久,所以大家多多支持啦


链表的结构及结构

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

 理解:可以把链表理解为火车,每一节火车车厢都有下一节车厢的钥匙

如下图

ee201a0eb16c480599aeda938fa4793c.png

 与顺序表不同的是:链表里面的‘车厢’都是独立申请下来的空间,我们称为节点

3b591deef0a84df2ae585eabc2119f04.png

如上图就可以理解为一个节点

当前节点主要有两个部分组成:要保存的数据和保存下一个节点的地址(指针变量)

为什么需要指针变量保存下一个节点的位置?

因为链表里面的每一个节点都是独立申请的,需要保存下一个节点的位置才能找到下一个节点

假设保存的节点为整型,则代码如图:

struct SListNode
{
int data; //节点数据
struct SListNode* next; //指针变量⽤保存下⼀个节点的地址
};

给定的链表结构中,如何实现链表的从头到尾的打印

void SLTPrint(SLTNode*phead)
{
 SLTNode*pcur=phead;
 while(pcur)
  {
       printf("%d ",pcur->date);
       pcur=pcur->next;
  } 
  printf("\n");
}

ee201a0eb16c480599aeda938fa4793c.png

 解析

1.pcur指针变量保存第一个节点的地址

2.对pcur解引用拿到next指针变量的地址一个节点的地址

3.赋值给pcur,此时的pcur保存的地址就指向了下一个节点

补充:

1、链式机构在逻辑上是连续的,在物理结构上不一定连续

2、节点一般是从堆上申请的

3、从堆上申请的空间,是按照一定策略分配的,每一次申请的空间可能连续 

 单链表的存储结构

先写出结构体改名比较简单的名字为SLTNode

​​​​typedef int SLTDataType;//便于后期修改
typedef struct SListNode
{
SLTDataType data; //节点数据
struct SListNode* next; //指针保存下⼀个节点的地址
}SLTNode;

单链表的头插

1、断言,pphead不能为空

2、给新节点申请空间

3、让新节点的next指向原本的头

4、新节点设为新的头

void SLTPushFront(SLTNode** pphead, SLTDateType x)
{
	assert(pphead);
	SLTNode* newnode = SLTBuyNode(x);
	newnode->next = *pphead;
	*pphead = newnode;
}

50148876f6a1493eb079942a815aacd2.png

单链表的尾插

1.分为链表为空和不为空

2.链表为空:头节点直接为新的节点,即newnode

3.链表不为空,找到尾节点,尾节点的next指向新节点newnode

注意:这里使用了二级指针,因为你需要传的是地址

//尾插
void SLTPushBack(SLTNode** pphead, SLTDateType x)
{
    assert(pphead);
	SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
	newnode->date = x;
	newnode->next = NULL;
	//链表为空
	if (*pphead == NULL)
	{
		*pphead = newnode;
		return;
	}
	//链表不为空
	SLTNode* ptail = *pphead;
	while (ptail->next)
	{
		ptail = ptail->next;
	}
	ptail->next = newnode;
}
void SListTest02()//尾插
{
	SLTNode* plist = NULL;
	SLTPushBack(&plist, 1);
	SLTPrint(plist);
}

单链表的头部删除

1、pphead,*pphead不能为空

2、保存第二个节点

3、释放旧的头也就是*pphead

4、把保存的第二个节点设为头结点

void SLTPopFront(SLTNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	//第二个节点成为新的头结点,释放旧的头结点
	SLTNode* next = (*pphead)->next;
	free(*pphead);
	*pphead = next;
}

单链表的尾部删除

1、分为两种情况,一个是链表只有一个节点的情况,另一个是不止有一个节点的情况,当然这都是在pphead *phead不为空的情况

2、只有一个节点的情况就是释放(free)

3、不只有一个节点:就是找到尾节点前面一个节点,让这个节点(prev)指向NULL找到尾节点,使用while循坏,把头结点保存在ptail中,当ptail->next为NULL就终止循坏,找到了prev    prev->next指向空指针,然后释放掉之前的尾节点

void SLTPopBack(SLTNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	//一个节点
	if ((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
		return;
	}
	SLTNode* ptail = *pphead;
	SLTNode* prev = NULL;
	while (ptail->next)
	{
		prev = ptail;
		ptail = ptail->next;
	}
	prev->next = NULL;
	//销毁
	free(ptail);
	ptail = NULL;
}

单链表的查找

1、断言pphead

2、遍历链表,采用循坏就可以,找到了就返回x

SLTNode* SLTFind(SLTNode** pphead, SLTDateType x)
{
	assert(pphead);
	//遍历链表
	SLTNode* pcur = *pphead;
	while (pcur)
	{
		if (pcur->date == x)
			return x;
		pcur = pcur->next;
	}
	return NULL;
}

在指定位置之前插入数据

1、pphead *pphead pos不能为空

2、分为两种情况,pos是头结点,采用头插

3、pos不是头结点,采用while循坏,找到pos前面的一个节点(prev),prev->next指向新节点,新节点->next指向pos

void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDateType x)
{
	assert(pphead);
	assert(pos);
	assert(*pphead);

	//Pos是头结点
	if (pos == *pphead)
	{
		SLTPushFront(pphead, x);
		return;
	}
	//不是头结点情况
	SLTNode* newnode = SLTBuyNode(x);
	SLTNode* prev = *pphead;
	
	while (prev->next != pos)
	{
		prev = prev->next;
	}
	prev->next = newnode;
	newnode->next = pos;
}

在指定位置之后插入数据

1、直接申请新节点空间

2、新节点的->next指向pos的后面一个节点

3、pos—>next指向新节点

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

3e6c113a20f24096a31f80644f359a9e.png

删除pos节点

1、pos是头结点,采用头删

2、pos不是头节点,找到pos节点的前面一个节点(prev),让prev->next指向pos->next,然后释放pos节点

void SLTErase(SLTNode** pphead, SLTNode*pos)
{
	assert(pphead);
	assert(*pphead);
	assert(pos);

	//posg刚好是头结点
	if (*pphead == pos)
	{
		SLTPopFront(pphead);
		return;
	}
	SLTNode* prev = *pphead;
	while (prev->next != pos)
	{
		prev = prev->next;
	}
	prev->next = pos->next;
	free(pos);
	pos = NULL;
}

销毁链表

采用循坏,依次把每一个节点释放掉

void SListDesTroy(SLTNode** pphead)
{
	assert(pphead);
	assert(*pphead);;

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


希望大家多多支持!谢谢大家可以阅读到这里^  ^

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

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

相关文章

在MongoDB建模1对N关系的基本方法

“我在 SQL 和规范化数据库方面拥有丰富的经验,但我只是 MongoDB 的初学者。如何建立一对 N 关系模型?” 这是我从参加 MongoDB 分享日活动的用户那里得到的最常见问题之一。 我对这个问题没有简短的答案,因为方法不只有一种,还有…

LangChain核心模块 Retrieval——文档加载器

Retrieval ​ 许多LLM申请需要用户的特定数据,这些数据不属于模型训练集的一部分,实现这一目标的主要方法是RAG(检索增强生成),在这个过程中,将检索外部数据,然后在执行生成步骤时将其传递给LLM。 ​ LangChain 提供…

探索设计模式的魅力:精准、快速、便捷:游标尺模式在软件设计中的三大优势

​🌈 个人主页:danci_ 🔥 系列专栏:《设计模式》 💪🏻 制定明确可量化的目标,并且坚持默默的做事。 精准、快速、便捷:游标尺模式在软件设计中的三大优势 文章目录 一、案例场景&…

黑马程序员:C++核心编程——3.函数提高

目录 1.函数默认参数 2.函数占位数 3.函数重载* 1.函数默认参数 形参列表中可以有默认值。 注意:如果某个位置有默认值,那么这个位置之后的都要有。如果函数声明有默认值了,函数实现的时候就不能有默认值(防止默认值不同而导…

蓝桥杯第十五届抱佛脚(二)竞赛中的数据结构

蓝桥杯第十五届抱佛脚(二)内置数据结构 文章目录 蓝桥杯第十五届抱佛脚(二)内置数据结构在竞赛中常见的数据结构数组(Array)链表(Linked List)栈(Stack)队列(Queue)树(Tree)映射(Map) 内置数据结构的快速使用迭代器(It…

综合知识篇20-基于中间件的开发新技术考点(2024年软考高级系统架构设计师冲刺知识点总结系列文章)

专栏系列文章: 2024高级系统架构设计师备考资料(高频考点&真题&经验)https://blog.csdn.net/seeker1994/category_12593400.html案例分析篇00-【历年案例分析真题考点汇总】与【专栏文章案例分析高频考点目录】(2024年软考高级系统架构设计师冲刺知识点总结-案例…

强化学习之父Richard Sutton:通往AGI的另一种可能

2019年,强化学习之父、阿尔伯塔大学教授Richard Sutton发表了后来被AI领域奉为经典的The Bitter lesson,这也是OpenAI研究员的必读文章。 在这篇文章中,Richard指出,过去 70 年来,AI 研究的一大教训是过于重视人类既有…

Python学习从0到1 day18 Python可视化基础综合案例 1.折线图

我默记这段路的酸楚,等来年春暖花开之时再赏心阅读 —— 24.3.24 python基础综合案例 数据可视化 — 折线图可视化 一、折线图案例 1.json数据格式 2.pyecharts模块介绍 3.pyecharts快速入门 4.数据处理 5.创建折线图 1.json数据格式 1.什么是json 2.掌握如何使用js…

用kimichat炒股:查找比亚迪、特斯拉等电动车产业链相关股票

kimichat可以联网检索,搜索结果更加准确、智能。 在kimichat中输入提示词:找出同时在比亚迪产业链和特斯拉产业链的企业 根据您提供的搜索结果中的信息,我们可以找出同时在比亚迪产业链和特斯拉产业链中的企业。以下是从搜索结果中提取的相关…

【动态规划】Leetcode 70. 爬楼梯

【动态规划】Leetcode 70. 爬楼梯 解法1 ---------------🎈🎈题目链接🎈🎈------------------- 假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢? 输入…

先进电机技术 —— 长线缆驱动电机面临哪些问题?

一、长线驱动问题简述 电机变频驱动器(VFD)输出侧采用长线缆驱动电机运行时,将会面对多种问题,主要包括但不限于: 此图片来源于网络 1. **电压降**: - 长线缆的电阻会导致电压降增大,当电…

智能优化算法 | Matlab实现牛顿-拉夫逊优化算法Newton-Raphson-based optimize(内含完整源码)

文章目录 效果一览文章概述源码设计参考资料效果一览 文章概述 智能优化算法 | Matlab实现牛顿-拉夫逊优化算法Newton-Raphson-based optimize(内含完整源码) 源码设计 % ------------------------------------------------------------------------------------------------…

UNI-APP读取本地JSON数据

首先要把json文件放在static文件夹下 然后在要读取数据的页面导入 import data from ../../static/data.json读取数据: onLoad() {console.log(data, data)}, 打印出来的就是JSON文件里的数据了

【SQL】1527. 患某种疾病的患者(like;通配符)

前述 知识点回顾: MySQL 使用OR在LIKE查询中比较多个字段 %:表示任意字符(包括0个或多个)_:表示任意单个字符匹配空格:直接用空格就行,例如,% DIAB1%可以匹配字符串ACNE DIAB100 …

利用免费 GPU 部署体验大型语言模型推理框架 vLLM

vLLM简介 vLLM 是一个快速且易于使用的 LLM(大型语言模型)推理和服务库。 vLLM 之所以快速,是因为: 最先进的服务吞吐量 通过 PagedAttention 高效管理注意力键和值内存 连续批处理传入请求 使用 CUDA/HIP 图快速模型执行 量…

瑞芯微RK3576|触觉智能:开启科技新篇章

更多产品详情可关注深圳触觉智能官网! “瑞芯微,创新不止步!”——全新芯片RK3576即将震撼登场。指引科技风潮,创造未来无限可能!这款芯片在瑞芯微不断创新和突破的道路上,不仅是对过往成就的完美延续&…

Python爬虫-批量爬取星巴克全国门店

前言 本文是该专栏的第22篇,后面会持续分享python爬虫干货知识,记得关注。 本文笔者以星巴克为例,通过Python实现批量爬取目标城市的门店数据以及全国的门店数据。 具体的详细思路以及代码实现逻辑,跟着笔者直接往下看正文详细内容。(附带完整代码) 正文 地址:aHR0cHM…

基于python+vue电影院订票信息管理系统flask-django-php-nodejs

根据此问题,研发一套电影院订票信息管理系统,既能够大大提高信息的检索、变更与维护的工作效率,也能够方便信息系统的管理运用,从而减少信息管理成本,提高效率。 该电影院订票信息管理系统采用B/S架构、前后端分离以及…

Nacos部署(二)Linux部署Nacos2.3.x集群环境

😊 作者: 一恍过去 💖 主页: https://blog.csdn.net/zhuocailing3390 🎊 社区: Java技术栈交流 🎉 主题: Nacos部署(二)Linux部署Nacos2.3.x集群环境 ⏱️…

隐私计算实训营第四讲-SecretFlow 环境安装与部署

SecretFlow 环境安装与部署 SecretFlow环境的安装和部署指南,包括仿真模式和生产模式的配置方法。 1. 环境安装 要安装SecretFlow环境,请按照以下步骤操作: 1.1 创建并激活Conda环境 创建名为 sf 的新Conda环境,并指定Python版…