数据结构入门 — 链表详解_单链表

news2025/1/9 10:56:19

前言

数据结构入门 — 单链表详解*
博客主页链接:https://blog.csdn.net/m0_74014525
关注博主,后期持续更新系列文章
文章末尾有源码
*****感谢观看,希望对你有所帮助*****


系列文章

第一篇:数据结构入门 — 链表详解_单链表
第二篇:数据结构入门 — 链表详解_双向链表
第三篇:数据结构入门 — 链表详解_循环链表


文章目录

  • 前言
  • 系列文章
  • 一、链表
    • 1. 链表是什么
    • 2. 优缺点
  • 二、概念及结构
    • 1. 概念
    • 2. 结构
  • 三、 链表的分类
    • 1. 链表结构类型
    • 2. 常用的两种链表结构
  • 四、单链表接口实现(代码演示)
    • 1. 动态申请一个结点
    • 2. 单链表打印
    • 3. 单链表增删查改
    • 4. 单链表销毁
  • 五、所有文件代码
    • 1. Gitee链接
  • 总结


一、链表

1. 链表是什么

链表是一种数据结构,由一系列节点组成,在每个节点中存储数据和指向下一个节点的指针。每个节点只包含指向下一个节点的指针,不像数组那样有预定义的大小。链表可以动态地增长和缩小,并且非常灵活,可以在任何位置插入或删除节点。链表主要分为单向链表、双向链表和循环链表等不同类型。

2. 优缺点

链表的优点:

  1. 灵活性:由于链表中每个节点都有指向下一个节点的指针,因此链表可以在任何位置添加或移除元素,使得链表非常灵活。
  2. 动态性:由于链表的大小不是固定的,当需要增加或删除元素时,内存中不需要重新分配空间,而是在链表中增加或删除一个结点。
  3. 无需占用连续的内存空间:相较于数组等数据结构,链表的每个元素之间并没有关联性,每个节点可以在内存中的任意位置,不需要占用连续的内存空间。

链表的缺点:

  1. 没有随机访问的能力:链表中的元素之间没有关联性,无法像数组那样通过下标直接访问某个元素,需要从头开始遍历整个链表才能找到需要的元素,这会导致链表的查找效率比数组低。
  2. 内存空间的额外开销:由于需要在每个节点中存储指向下一个节点的指针,链表内每个元素需要占用比其存储内容更多的内存空间,这会导致链表没有数组那么节省内存。
  3. 不支持常量时间内的随机访问:链表只能在头尾节点进行快速的插入和删除操作,但在其他位置插入和删除节点较为困难,需要先遍历找到要操作的位置,这会导致链表操作的时间复杂度较高。

二、概念及结构

1. 概念

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

2. 结构

在现实的数据结构中,链表的结构
在这里插入图片描述
注意:

  1. 从上图可看出,链式结构在逻辑上是连续的,但是在物理上不一定连续
  2. 现实中的结点一般都是从堆上申请出来的
  3. 从堆上申请的空间,是按照一定的策略来分配的,两次申请的空间可能连续,也可能不连续

三、 链表的分类

1. 链表结构类型

链表可以分为单链表、双向链表和循环链表三种类型。

类型介绍
单链表节点只包含一个指向后继节点的指针,每个节点只知道下一个节点的位置。
双向链表节点包含一个指向前驱节点和一个指向后继节点的指针,每个节点都知道前面和后面的节点位置。
循环链表链表的最后一个节点指向第一个节点,形成一个环形结构。

除了这三种基本类型,还有一些派生的链表结构,如带有头节点的链表、带有尾指针的链表、带有哨兵节点的链表等。如下图有8种链表结构
1. 单向或者双向
在这里插入图片描述

2. 带头或者不带头(哨兵位)
在这里插入图片描述

3. 循环或者不循环

在这里插入图片描述

2. 常用的两种链表结构

无头单向非循环链表:

在这里插入图片描述
无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结
构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。

带头双向循环链表:
在这里插入图片描述
带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都
是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带
来很多优势,实现反而简单了,后面我们代码实现了就知道了。


四、单链表接口实现(代码演示)

无头+单向+非循环链表增删查改实现

1. 动态申请一个结点

SLTNode* BuySListNode(SLTDataType x)
{
	SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
	if (newnode == NULL)
	{
		perror("BuySListNode");
		exit(-1);
	}
	newnode->data = x;
	newnode->next = NULL;
	return newnode;
}

2. 单链表打印

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

3. 单链表增删查改

头插:

//头插
void SLTPushFront(SLTNode** pphead, SLTDataType x)
{

	assert(pphead);
	SLTNode* newnode = BuySListNode(x);
	newnode->next = *pphead;
	*pphead = newnode;
}

尾插:

//尾插
void SLTPushBack(SLTNode** pphead, SLTDataType x)
{

	assert(pphead);
	SLTNode* newnode = BuySListNode(x);
	
	//如果为空 
	if (*pphead==NULL)
	{
		*pphead = newnode;
	}
	else
	{ 
		SLTNode* tail = *pphead;
		while (tail->next != NULL)
		{
			tail = tail->next;
		}
		tail->next = newnode;
	}
}

头删:

//头删
void SLTPopFront(SLTNode** pphead)
{

	//空
	assert(pphead);
	assert(*pphead);
	//非空
	SLTNode* newnode = (*pphead)->next;
	free(*pphead);
	*pphead = newnode;

}

尾删:

//尾删
void SLTPopBack(SLTNode** pphead)
{
	assert(*pphead);
	assert(pphead);

   //一个节点
	if((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;

	}
	else
	{
		SLTNode* tail = *pphead;
		SLTNode* tailPrev = NULL;
		while(tail->next != NULL)
		{
			tailPrev = tail;
			tail = tail->next;
		}

		free(tail);
		tail = NULL;
		tailPrev->next = NULL;
	}
}

查找:

//查找
SLTNode* SLTFind(SLTNode* phead, SLTDataType x)
{
	SLTNode* cur = phead;
	while (cur)
	{
		if (cur->data == x)
		{
			return cur;
		}
		cur = cur->next;
	}

	return NULL;
}

在pos之前插入x:

// 在pos之前插入x
void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{

	assert(pos);
	assert(pphead);
	if (pos == *pphead)
	{
		SLTPushFront(pphead,x);
	}
	else
	{
		SLTNode* prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		SLTNode* newnode = BuySListNode(x);

		prev->next = newnode;
		newnode->next = pos;

	}

}

在pos以后插入x:

// 在pos以后插入x
void SLTInsertAfter(SLTNode* pos, SLTDataType x)
{
	assert(pos);
	SLTNode* newnode = BuySListNode(x);
	newnode->next = pos->next;
	pos->next = newnode;
	
}

删除pos位置:

// 删除pos位置
void SLTErase(SLTNode** pphead, SLTNode* pos)
{
	assert(pos);
	assert(pphead);
	if (pos == *pphead)
	{
		SLTPopFront(pphead);
	}
	else
	{
		SLTNode* tail = *pphead;
		while (tail->next != pos)
		{
			tail = tail->next;
		}
		tail->next = pos->next;
		free(pos);
		pos = NULL;
	}
}

删除pos的后一个位置:

// 删除pos的后一个位置
void SLTEraseAfter(SLTNode* pos)
{
	assert(pos);
	assert(pos->next);

    SLTNode* posNext = pos->next;

	pos->next = posNext->next;

	pos->next = posNext->next;

	free(posNext);

	posNext = NULL;

}

4. 单链表销毁

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

	SLTNode* cur = *pphead;
	while (cur)
	{
		SLTNode* del = cur->next;

		free(cur);

		cur = del;
	}

	
	* pphead == NULL;
}

五、所有文件代码

1. Gitee链接

***查看所有代码***
点击右边蓝色文字 DuckBro Gitee 代码仓库


总结

单链表的重点包括:

  1. 定义:单链表是一种数据结构,它由一系列节点组成,每个节点包含数据和指向下一个节点的指针。
  2. 插入操作:单链表的插入操作需要先找到要插入位置的前一个节点,然后将新节点插入到其后。
  3. 删除操作:单链表的删除操作需要先找到要删除的节点的前一个节点,然后将其指针指向下一个节点即可。
  4. 遍历操作:单链表需要从头节点开始依次遍历各个节点,获取其中存储的数据。
  5. 链表反转:单链表的反转操作是将链表中的节点从后往前连接,最后将原来的头节点变为尾节点。
  6. 快慢指针:快慢指针是单链表中常用的技巧,可以用于查找链表中的中间节点、判断是否有环等操作。
  7. 环形链表:有些单链表可能存在环形结构,即某个节点的指针指向之前的某个节点,使用快慢指针可以判断链表是否为环形。
  8. 链表排序:单链表可以使用排序算法进行排序,如冒泡排序、快速排序等。
  9. 链表的应用:单链表广泛应用于各种数据结构和算法中,如哈希表、图等。

如这篇博客对大家有帮助的话,希望 三连 支持一下 !!! 如果有错误感谢大佬的斧正 如有 其他见解发到评论区,一起学习 一起进步。

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

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

相关文章

OS 用户级线程

这个程序执行到D,Yield()跳到204,然后出栈404出错

2023 ccpc 网络赛 L 题解

Problem L. Partially Free Meal 题面 官方题解 官方题解解读 w(k,x)计算部分 主席树常规做法&#xff0c;在一般主席树中多维护一个这个区间的总和就ok了 根据单调性分治求解部分 接下来我们来推导一下单调性&#xff0c;也就是题解中的 f ( 1 ) < f ( 2 ) < f ( …

六、pikachu之目录遍历

文章目录 一、目录遍历概述二、目录遍历实战 一、目录遍历概述 在web功能设计中&#xff0c;很多时候我们会要将需要访问的文件定义成变量&#xff0c;从而让前端的功能便的更加灵活。 当用户发起一个前端的请求时&#xff0c;便会将请求的这个文件的值&#xff08;比如文件名…

鸡肋的RDP反制

更新时间&#xff1a;2023年07月19日09:18:29 为什么叫鸡肋&#xff1a; 鸡肋者&#xff0c;食之无肉&#xff0c;弃之有味。 你说不能成吧&#xff0c;但是有成功案例&#xff0c;你说成了吧&#xff0c;要求太高&#xff0c;还要看运气的。 一句话&#xff1a;对方需要开启…

如何通过PreMaint状态监测系统发现设备故障?

在现代工业领域&#xff0c;设备故障可能导致严重的生产中断和高额维修成本。然而&#xff0c;通过新兴的技术&#xff0c;如基于状态的维修&#xff0c;我们可以更早地发现故障并采取预防措施&#xff0c;从而实现生产的连续性和成本的降低。其中&#xff0c;PreMaint状态监测…

老板电器:劲草式企业才能穿越周期

材料力学上有两个概念&#xff1a;一个是韧性&#xff0c;用来形容材料在塑性变形和破裂过程中吸收能量的能力&#xff0c;韧性越好&#xff0c;发生脆性断裂的可能性越小&#xff1b;一个是塑性&#xff0c;是材料所受外力超过其弹性限度后具有的能永久保留形变的性质。 这两…

X2000 Linux UVC

参考文档&#xff1a;\doc\开发使用说明\USB使用说明文档\设备\USB_UVC\xburst2\USB_UVC.pdf 一、内核添加USB UVC功能 1、确定所用dts文件 进入到/tools/iconfigtool/IConfigToolApp/路径下&#xff0c;执行./IConfigTool 选择config文件&#xff0c;查看kernel默认配置 配…

油烟监管云平台在油烟净化技术中的应用研究

安科瑞 华楠 摘 要&#xff1a;介绍了生活中餐饮业油烟废气的排放特点、产生过程及其对人体和环境带来的环境&#xff0c;分析了餐饮油烟组成成分的成分特性及相关的致病症状。油烟净化器仍然是目前治理餐饮油烟比较有用的方法&#xff0c;基于高压静电原理的复合型油烟净化技…

前端面试手撕编程

目录 ES6 改变this call typeof this ! function context context || window context._this this delete context._this bind&#xff1a; return _this.apply(context, [...arguments].slice(1)); 深拷贝 !arr|| arr null || typeof arr ! object arr instanceof…

正则中常见的流派及其特性

目前正则表达式主要有两大流派&#xff08;Flavor&#xff09;&#xff1a;POSIX 流派与 PCRE 流派。 1、 POSIX 流派 POSIX 规范定义了正则表达式的两种标准&#xff1a; BRE 标准&#xff08;Basic Regular Expression 基本正则表达式&#xff09;&#xff1b;ERE 标准&am…

Android获取手机已安装应用列表JAVA实现

最终效果: 设计 实现java代码: //获取包列表private List<String> getPkgList() {List<String> packages new ArrayList<String>();try {//使用命令行方式获取包列表Process p Runtime.getRuntime().exec("pm list packages");//取得命令行输出…

【微信小程序开发】一文带你详解小程序组件和 API 的使用

引言 在小程序开发中&#xff0c;组件和API是非常重要的部分&#xff0c;它们可以帮助我们构建丰富的用户界面和实现各种功能。本文将介绍小程序中常用的组件和API&#xff0c;并提供相应的代码示例。 文章目录 引言组件文本组件图片组件按钮组件输入框组件列表组件 API网络请…

C语言小练习(三)

&#x1f31e; “也许你感觉自己与周遭格格不入&#xff0c;但正是那些你一人度过的时光&#xff0c;让你变得越来越有意思&#xff0c;等有天别人终于注意到你的时候&#xff0c;他们就会发现一个比他们想象中更酷的人。”-《生活大爆炸》 Day03 &#x1f4dd; 一.选择题&…

【Vue框架】基本的login登录

前言 最近事情比较多&#xff0c;只能抽时间看了&#xff0c;放几天就把之前弄的都忘了&#xff0c;现在只挑着核心的部分看。现在铺垫了这么久&#xff0c;终于可以看前端最基本的登录了&#x1f602;。 1、views\login\index.vue 由于代码比较长&#xff0c;这里将vue和js…

vxe-table解决虚拟滚动时的白屏问题

项目中使用虚拟滚动&#xff0c;滚动一快就会白屏&#xff0c;花了两天时间看了源码&#xff0c;检查渲染元素&#xff0c;发现元素渲染的是很快的&#xff0c;跟得上的&#xff0c;但是还是会出现白屏&#xff0c;后面检查样式&#xff0c;发现只要把表格背景色去掉&#xff0…

平安私人银行慈善沙龙广州站:善财传承公益有道,广州分行聚爱同行

近年来&#xff0c;平安私人银行将慈善作为客户服务的王牌权益之一&#xff0c;激发和满足客户公益慈善心愿&#xff0c;打造财富人群和困境人群的桥梁&#xff0c;并链接公益机构等专业组织&#xff0c;深度挖掘金融赋能慈善的多种可能性&#xff0c;让财富通过慈善事业释放出…

STM32CubeMX配置STM32G0 Standby模式停止IWDG(HAL库开发)

1.打开STM32CubeMX选择好对应的芯片&#xff0c;打开IWDG 2.打开串口1进行调试 3.配置好时钟 4.写好项目名称&#xff0c;选好开发环境&#xff0c;最后获取代码。 5.打开工程&#xff0c;点击魔术棒&#xff0c;勾选Use Micro LIB 6.修改main.c #include "main.h"…

springCloud整合Zookeeper的时候调用找不到服务

SpringCloud整合Zookeeper的时候调用找不到服务 首先&#xff0c;我们在注册中心注册了这个服务&#xff1a; 然后我们使用RestTemplate 调用的时候发现失败了&#xff1a;找不到这个服务&#xff1a; 找了很多资料发现这个必须要加上负载才行 BeanLoadBalanced //负载publi…

javaee idea创建maven项目,使用el和jstl

如果使用el表达式出现下图问题 解决办法 这是因为maven创建项目时&#xff0c;web.xml头部声明默认是2.3&#xff0c;这个默认jsp关闭el表达式 办法1 在每个需要用到el和jstl的页面的上面加一句: <% page isELIgnored"false" %> 方法2 修改web.xml文件开…

node和前端项目宝塔部署

首先需要一台服务器 购买渠道&#xff1a;阿里云、腾讯云、百度云、华为云 一、以阿里云为例 购买esc 可临时购买测试服务器 二、安装宝塔 复制公网ip地址 通过Xshell 进行账号密码的连接 连接后访问宝塔官网 宝塔面板下载&#xff0c;免费全能的服务器运维软件 找到自己…