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

数据结构:单链表

文章目录

  • 1. 单链表的概念及结构
  • 2. 单链表相关操作
    • 2.1 创建节点
    • 2.2 尾插
    • 2.3 打印
    • 2.4 头插
    • 2.5 尾删
    • 2.6 头删
    • 2.7 查找
    • 2.8 指定位置后插入
    • 2.9 指定位置前插入
    • 2.10 删除指定位置
    • 2.11 删除指定位置后的节点
    • 2.12 销毁单链表
  • 3. 链表种类

在这里插入图片描述

1. 单链表的概念及结构

概念:链表是⼀种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的。
在这里插入图片描述
链表的结构跟火车车厢相似,淡季时车次的车厢会相应减少,旺季时车次的车厢会额外增加几节。只需要将火车里的某节车厢去掉/加上,不会影响其他车厢,每节车厢都是独立存在的。

与顺序表不同的是,链表里的每节"车厢"都是独立申请下来的空间,我们称之为“结点/节点”。节点的组成主要有两个部分:当前节点要保存的数据和保存下⼀个节点的地址(指针变量)

为什么还需要指针变量来保存下⼀个节点的位置?
链表中每个节点都是独立申请的(即需要插入数据时才去申请⼀块节点的空间),我们需要通过指针变量来保存下⼀个节点位置才能从当前节点找到下⼀个节点。

因此链表的结构我们可以这样设计:

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

注意:

  • 链式结构在逻辑上是连续的,在物理结构上不⼀定连续
  • 节点⼀般是从堆上申请的
  • 从堆上申请来的空间,是按照⼀定策略分配出来的,每次申请的空间可能连续,可能不连续

2. 单链表相关操作

2.1 创建节点

//创建节点
SNode* BuyNode(SLDatatype x)
{
	SNode* newnode = (SNode*)malloc(sizeof(SNode));
	if (!newnode)
	{
		perror("malloc");
		return NULL;
	}
	newnode->data = x;
	newnode->next = NULL;

	return newnode;
}

2.2 尾插

  • 若当前链表尾空,则插入节点就是头节点
  • 不为空,先找尾,再插入
//尾插
void SListPushBack(SNode** pphead, SLDatatype x)
{
	assert(pphead);

	SNode* newnode = BuyNode(x);
	//链表为空
	if (*pphead == NULL)
	{
		*pphead = newnode;
		return;
	}
	SNode* tail = *pphead;
	//找尾
	while (tail->next)
	{
		tail = tail->next;
	}
	//连接
	tail->next = newnode;
}

2.3 打印

//打印
void SListPrint(SNode* phead)
{
	SNode* cur = phead;
	while (cur)
	{
		printf("%d->", cur->data);
		cur = cur->next;
	}
	printf("NULL\n");
}

在这里插入图片描述

2.4 头插

  • 先将新节点指向头节点
  • 新节点变为新的头节点
//头插
void SlistPushFront(SNode** pphead, SLDatatype x)
{
	assert(pphead);
	SNode* newnode = BuyNode(x);

	newnode->next = *pphead;
	*pphead = newnode;
}

2.5 尾删

  • 链表不能为空
  • 找尾节点,同时记录其前驱节点
    • 若只有头节点,则将头节点置空
    • 多个节点,释放尾节点,将其前驱节点的next置空
void SlistPopBack(SNode** pphead)
{
	assert(pphead);
	//链表不为空
	assert(*pphead);
	//只有一个节点
	if ((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
		return;
	}
	//多个节点
	SNode* cur = *pphead;
	SNode* prev = NULL;
	while(cur->next)
	{
		prev = cur;
		cur = cur->next;
	}
	//释放尾节点
	free(cur);
	cur = NULL;
	prev->next = NULL;
}

在这里插入图片描述

2.6 头删

  • cur记录当前头节点,头节点后移
  • 释放cur
void SlistPopFront(SNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	SNode* cur = *pphead;
	
	*pphead = (*pphead)->next;
	free(cur);
	cur = NULL;
}

在这里插入图片描述

2.7 查找

找到目标,返回指向该目标的指针

//查找
SNode* SListFind(SNode** pphead, SLDatatype x)
{
	assert(pphead);
	SNode* cur = *pphead;
	while (cur)
	{
		if (cur->data == x)
		{
			return cur;
		}
		cur = cur->next;
	}
	return NULL;
}

2.8 指定位置后插入

  • 将新节点与pos的后继节点连接
  • pos节点指向新节点
//指定位置后插入
void SListInsertAfter(SNode* pos, SLDatatype x)
{
	assert(pos);
	SNode* newnode = BuyNode(x);
	newnode->next = pos->next;
	pos->next = newnode;
}

2.9 指定位置前插入

  • 若指定位置为头节点,则选择头插
  • 找到pos位置的前驱节点,与新节点相连
void SListInsertBefore(SNode** pphead, SNode* pos, SLDatatype x)
{
	assert(pphead);
	assert(pos);
	//指定位置为头节点,执行头插
	if (pos == *pphead)
	{
		SlistPushFront(pphead, x);
		return;
	}
	//找pos位置的前驱
	SNode* cur = *pphead;
	while (cur->next != pos)
	{
		cur = cur->next;
	}
	SNode* newnode = BuyNode(x);
	cur->next = newnode;
	newnode->next = pos;
}

在这里插入图片描述

2.10 删除指定位置

  • 若指定位置为头节点,执行头删
  • 前驱节点指向后继节点,释放当前节点
void SListDelePos(SNode** pphead, SNode* pos)
{
	assert(pphead);
	assert(pos);
	//指定位置为头节点,执行头删
	if (pos == *pphead)
	{
		SlistPopFront(pphead);
		return;
	}
	SNode* cur = *pphead;
	while (cur->next != pos)
	{
		cur = cur->next;
	}
	cur->next = pos->next;
	free(pos);
	pos = NULL;
}

在这里插入图片描述

2.11 删除指定位置后的节点

  • 指定位置后必须有节点
  • 记录指定位置后的节点
  • pos指向指定位置后的后继节点
  • 释放指定位置后的节点
void SListDeleAfter(SNode* pos)
{
	assert(pos);
	//后面得有元素
	assert(pos->next);
	SNode* dele = pos->next;
	pos->next = dele->next;
	free(dele);
	dele = NULL;
}

2.12 销毁单链表

  • 先记录后继节点,再销毁当前节点
void SListDestroy(SNode** pphead)
{
	assert(pphead);
	assert(*pphead);

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

3. 链表种类

链表的结构非常多样,以下情况组合起来就有8种(2x2x2)链表结构:
在这里插入图片描述

在这里插入图片描述
虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构: 单链表双向带头循环链表

  1. 无头单向非循环链表:结构简单,⼀般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试⾯试中出现很多。
  2. 带头双向循环链表:结构最复杂,⼀般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了

http://www.kler.cn/a/229812.html

相关文章:

  • 预训练语言模型——BERT
  • oracle闪回表
  • 数据集-目标检测系列- 电话 测数据集 call_phone >> DataBall
  • 基于 OPENCV 和 MFC 的图像处理程序
  • Three.js 渲染技术:打造逼真3D体验的幕后功臣
  • kubeneters-循序渐进Cilium网络(二)
  • LeetCode 0292.Nim 游戏:脑筋急转弯
  • 【经典例子】Java实现2048小游戏(附带源码)
  • 【自然语言处理-工具篇】spaCy<1>--介绍及安装指南
  • 8个国外顶尖设计网站,设计师必备!
  • re:从0开始的CSS学习之路 2. 选择器超长大合集
  • Java锁到底是个什么东西
  • 92.网游逆向分析与插件开发-游戏窗口化助手-显示游戏数据到小助手UI
  • 12. onnx转为rknn测试时有很多重叠框的修改(python)
  • Bug地狱 #1 突然宕机,企业级应用到底怎么了
  • Android 无操作之后定时退出
  • 代码随想录算法训练营第二八天 | 分割 子集
  • Python 调用 OpenAI ChatGPT API
  • leetcode-top100链表专题二
  • Django通过Json配置文件分配多个定时任务
  • 比较两次从接口获取的数据,并找出变动的字段
  • 071:vue中过滤器filters的使用方法(图文示例)
  • Z函数的原理和应用:以Python为例
  • 微信自动预约小程序开发指南:从小白到专家
  • HiSilicon352 android9.0 开机视频调试分析
  • Micro micro controller一览