【数据结构】单链表的层层实现!! !

这篇具有很好参考价值的文章主要介绍了【数据结构】单链表的层层实现!! !。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【数据结构】单链表的层层实现!! !,数据结构,c语言
关注小庄 顿顿解馋(●’◡’●)

上篇回顾
我们上篇学习了本质为数组的数据结构—顺序表,顺序表支持下标随机访问而且高速缓存命中率高,然而可能造成空间的浪费,同时增加数据时多次移动会造成效率低下,那有什么解决之法呢?这就得引入我们链表这种数据结构

一.何为链表

🏠 链表概念

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

特点:物理结构不一定连续,逻辑结构连续
【数据结构】单链表的层层实现!! !,数据结构,c语言
我们的链表结构类似我们的火车,有头有尾,中间每个结点被有序链接;与火车不同的是,链表的结点可能不是紧挨着的。

【数据结构】单链表的层层实现!! !,数据结构,c语言

类似这样,我们可以得出:
1.每个结点由数据和下一结点地址两部分组成,而每个结点构成了一个链表。
2.每个结点保存的是下一个结点的地址,这样就能找到下一个结点,最后为空就停止
3.每个结点的地址不是连续的,可以体现出链表的物理结构不一定连续

注:我们的结点一般是在堆区开辟的,因为此时你在程序结束前不free就会一直存在这块空间,同时可根据需要灵活申请结点存数据。

这样我们就可以用一个结构体封装每个结点:

typedef int Datatype;
typedef struct ListNode
{
	Datatype x;
	struct ListNode* next;
}Node;

注: 这里我们可以用typedef来重命名我们要存储的数据类型,这样对于不同数据的操作我们只要改typedef即可。

🏠 链表的分类

我们根据链表三个特点:1.带头不带头 2.单向还是双向 3.循环还是不循环 组合成了如上的8种链表
本篇博客,我们要实现的是单向不带头不循环链表(单链表),至于什么是带头,双向,循环我们下回双链表再进行讲解


二.单链表的实现

无头+单向+不循环链表的增删差改

🏠 链表的打印

  • 链表数据的打印

这个接口就很好的体现了结点结构保存指针的妙处了~

//链表的打印
void SLTPrint(Node* phead)
{
	asser(phead);
	Node* cur = phead;
	while (cur)
	{
		printf("%d ", cur->x);
		cur = cur->next;
	}
}

🏠 链表的头插和尾插

  • 链表的尾插

【数据结构】单链表的层层实现!! !,数据结构,c语言
对于链表的尾插要注意几个问题:1.申请新结点 2.链表是否为空

解决方法:1.对于链表为空,直接让申请的新节点作为头节点 2.对于不为空的链表,首先要找到尾结点,再进行插入 3.对于申请新节点,我们后续的头插也要使用我们可以封装成一个接口,同时结点在堆区申请,调用完接口就不会释放了。

//申请新节点
Node* BuyNode(Datatype x)
{
	Node* newnode = (Node*)malloc(sizeof(Node));
	if (newnode == NULL)
	{
		perror("malloc failed");
		return;
	}
	newnode->next = NULL;
	newnode->x = x;
	return newnode;
}

void SLTPushBack(Node** pphead, Datatype x)
{
	assert(pphead);
	//申请新节点
	Node* newnode = BuyNode(x);
	//链表为空
	if (NULL == *pphead)
	{
		*pphead = newnode;
		return;
	}
	//链表不为空:1.找尾巴结点2.插入新节点
	Node* ptail = *pphead;
	while (ptail->next)
	{
		ptail = ptail->next;
	}
	ptail->next = newnode;
}

思考:这里为什么传二级指针?如果不传二级指针呢?

void SLTNPushBack(Node* pphead, Datatype x)
{
	
	//申请新节点
	Node* newnode = BuyNode(x);
	//链表为空
	if (NULL == pphead)
	{
		pphead = newnode;
		return;
	}
	//链表不为空:1.找尾巴结点2.插入
	Node* ptail = pphead;
	while (ptail->next)
	{
		ptail = ptail->next;
	}
	ptail->next = newnode;
}
int main()
{
	Node* n1 = NULL;
	SLTNPushBack(n1,1);
	return 0;
}

【数据结构】单链表的层层实现!! !,数据结构,c语言
经过观察传一级指针版本的调用前后,我们发现n1这个指针变量存储的值并没发生改变,究其原因如下图
【数据结构】单链表的层层实现!! !,数据结构,c语言

  • 链表的头插
    ,【数据结构】单链表的层层实现!! !,数据结构,c语言
    *对于链表的头插要注意的问题:1.申请新节点 2.链表释放为空 *

解决方法:1.链表为空时,申请的新结点作为头结点 2.链表不为空时,让newnode->next指向原来头节点,再让newnode成为新的头节点。

void SLTPushFront(Node** pphead, Datatype x)
{
	assert(pphead);
	//申请新节点
	Node* newnode = BuyNode(x);
	//链表为空
	if (NULL == *pphead)
	{
		*pphead = newnode;
		return;
	}
	newnode->next = *pphead;
	*pphead = newnode;
}

🏠 链表的尾删和头删

  • 链表的尾删
    【数据结构】单链表的层层实现!! !,数据结构,c语言
    对于链表的尾删,我们需要分三种情况!

1.链表为空时此时删不了直接退出
2.链表只有一个结点时,释放头节点,置phead为空
3.链表有多个结点时,我们需要遍历链表找到尾结点的前置结点,先释放尾节点再将前置结点的next置为空
注:不能先将前置结点的next置为空,再释放尾结点,此时就找不到尾结点的地址了。

//链表的尾删和头删
void SLTPopBack(Node** pphead)
{
	assert(pphead);
	assert(*pphead);//判断链表不为空
	if ((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
		return;
	}
	Node* pre = *pphead;
	while (pre->next->next)
	{
		pre = pre->next;
	}
	free(pre->next);
	pre->next = NULL;
}
  • 链表的头删

【数据结构】单链表的层层实现!! !,数据结构,c语言
对于链表的头删,分为两种情况就可以了,因为有了phead很方便~

1.链表为空时,删不了直接断言下
2.链表不为空时,记录头节点的下一个位置,先释放头节点,再更换phead指向的位置

注:这里也不能先更新phead再释放头结点

void SLTPopFront(Node** pphead)
{
	assert(pphead);
	assert(*pphead);
	Node* pNext = (*pphead)->next;
	free(*pphead);
	*pphead = pNext;
}

🏠 链表指定位置的插入和删除

【数据结构】单链表的层层实现!! !,数据结构,c语言

1.链表为空时,无法插入
2.pos位置结点刚好是头结点,直接头插或头删
3.pos位置结点不是头节点,需要找到pos位置的前置结点,记录位置。

void NodeInpos(Node** pphead, Node* pos, Datatype x)
{
	assert(pphead);
	//pos不为空 -》 链表一定不能为空
	assert(pos);
	assert(*pphead);
	//建立一个新节点
	Node* newnode = BuyNode(x);
	//pos刚好是头结点的情况
	if (*pphead == pos)
	{
		//运用头插
		NodeinFront(pphead,x);
		return;
	}
	//pos刚好不是头结点
	//1.先找出pos前面的结点pre  2.newnode->next = pre->next 3.pre-<next = newnode
	Node* pre = *pphead;
	while (pre->next != pos)
	{
		pre = pre->next;
	}
	newnode->next = pre->next;
	pre->next = newnode;
}

void NodeDelpos(Node** pphead, Node* pos)
{
	assert(pphead);
	assert(pos);
	assert(*pphead);
	//pos刚好是第一个结点 执行头删
	if (*pphead == pos)
	{
		NodeDelFront(pphead);
		return;
	}
	//pos不是第一个结点 1.先找到那个pos前面结点pre 2.pre->next = pos->next 3.free
	Node* pre = *pphead;
	while (pre->next != pos)
	{
		pre = pre->next;
	}
	pre->next = pos->next;
	free(pos);
	pos = NULL;
}

🏠 链表的查找

Node* NodeFind(Node** phead, Datatype x)
{
	assert(phead);
	//遍历链表
	Node* pcur = *phead;
	while (pcur)
	{
		if (pcur->data == x)
		{
			return pcur;
		}
		pcur = pcur->next;
	}
	//找不到则返回NULL
}

这里建议用一个临时变量来遍历~

🏠 链表的销毁

void NodeDestroy(Node** pphead)
{
	assert(pphead);
	assert(*pphead);
	//1.创一个临时变量存pcur->next的地址
	Node* pcur = *pphead;
	while (pcur)
	{
		Node* next = pcur->next;
		free(pcur);
		pcur = next;
	}
	*pphead = NULL;
}

注: 这里要保存好下一个结点地址,销毁后就能继续遍历

三.单链表的分析以及与顺序表的比较

🏠 单链表的优缺点

通过实现我们的单链表,我们发现单链表有以下优点

1.单链表不存在空间浪费(根据需求灵活在堆上申请新结点)
2.单链表的任意插入和删除效率高
(分析: 这里是已经确定插入的位置,对于链表只需改变指针的指向就能实现插入和删除,时间复杂度是O(1),而数组插入和删除需要遍历数组,时间复杂度是O(N)

单链表也不是万能的,它在一些应用场景也发挥不出来作用

1.不支持随机访问
2.缓存命中率低
3.查找效率低

总结:链表适用于频繁任意插入和删除的场景,不适用于随机访问和查找

🏠 单链表与顺序表的比较

单链表 顺序表
物理空间不一定连续 物理空间一定连续
不支持随机访问 支持下标随机访问
插入和删除效率高 O(1) 插入和删除效率低 O(N)
缓存命中率低 缓存命中率高
无空间的浪费 可能造成数据丢失和空间浪费
缓存利用率高

综上 我们可以根据场景需求选择不同的结构,灵活运用数据~


本次分享到这结束了,下篇我们将讲解双向链表,不妨来个一键三连捏文章来源地址https://www.toymoban.com/news/detail-838963.html

到了这里,关于【数据结构】单链表的层层实现!! !的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • [C语言][数据结构][链表] 单链表的从零实现!

    目录 零.必备知识 1.一级指针 二级指针 2. 节点的成员列表     a.数据     b.指向下一个节点的指针. 3. 动态内存空间的开辟 (malloc-calloc-realloc) 一.单链表的实现与销毁          1.1 节点的定义         1.2 单链表的尾插         1.3 单链表的头插         1.4 单链表的尾删  

    2024年04月11日
    浏览(76)
  • 【数据结构】单链表的基本操作 (C语言版)

    目录 一、单链表 1、单链表的定义: 2、单链表的优缺点: 二、单链表的基本操作算法(C语言) 1、宏定义 2、创建结构体 3、初始化 4、插入 4、求长度 5、清空 6、销毁  7、取值 8、查找 9、删除 10、头插法创建单链表 11、尾插法创建单链表 三、单链表的全部代码(C语言)

    2024年01月22日
    浏览(60)
  • 【数据结构】 循环单链表的基本操作 (C语言版)

    目录 一、循环单链表 1、循环单链表的定义: 2、循环单链表的优缺点: 二、循环单链表的基本操作算法(C语言)    1、宏定义  2、创建结构体 3、循环单链表的初始化  4、循环单链表的插入 5、求单链表长度 6、循环单链表的清空 7、循环单链表的销毁 8、循环单链表的取

    2024年01月22日
    浏览(60)
  • 【数据结构】单链表的实现

    🌇个人主页:平凡的小苏 📚学习格言:别人可以拷贝我的模式,但不能拷贝我不断往前的激情 🛸C语言专栏:https://blog.csdn.net/vhhhbb/category_12174730.html 🚀数据结构专栏:https://blog.csdn.net/vhhhbb/category_12211053.html         家人们更新不易,你们的👍点赞👍和⭐关注⭐真的对我

    2023年04月09日
    浏览(65)
  • 【(数据结构)— 单链表的实现】

    概念: 链表是⼀种 物理存储结构上非连续、 非顺序的存储结构,数据元素的 逻辑顺序 是通过链表中的 指针链接 次序实现的 。 链表的结构跟⽕⻋⻋厢相似,淡季时⻋次的⻋厢会相应减少,旺季时⻋次的⻋厢会额外增加⼏节。只需要将⽕⻋⾥的某节⻋厢去掉/加上,不会影响

    2024年02月08日
    浏览(51)
  • 【数据结构】-- 单链表的实现

    在前面我们学习了顺序表,顺序表在数组的基础上提供了很多现成的方法,方便了我们对数据的管理,但是我们也发现顺序表有着许多不足: 在处理大型的数据时,需要频繁的增容且在中间删除或插入数据时需要遍历顺序表,这些性质导致了顺序表的 效率较低 。这时我们就

    2024年04月27日
    浏览(52)
  • 【数据结构—单链表的实现】

    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 目录 前言 1. 链表的概念及结构 2. 单链表的实现 2.1单链表头文件——功能函数的定义 2.2单链表源文件——功能函数的实现 2.3 单链表源文件——功能的测试 3.具体的理解操作图 4. 链表的分类 总结 世上

    2024年02月05日
    浏览(66)
  • C语言简单的数据结构:单链表的有关算法题(2)

    接着我们介绍后面的三道题,虽然代码变多了但我们的思路更加通顺了 题目链接:https://leetcode.cn/problems/merge-two-sorted-lists/ 创建新链表,遍历原链表,将节点值小的进行尾插到新链表中 这里要多次进行对NULL的判断,开始传入列表,中间newHead的判断,循环出来一个为NULL的判断

    2024年04月15日
    浏览(64)
  • 【数据结构】单链表的简单实现

    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 单链表是一种链式存取的数据结构,用一组地址任意的存储单元存放线性表中的数据元素。链表中的数据是以结点来表示的,每个结点的构成:元素(数据元素的映象) + 指针(指示后继元素存储位置),元

    2024年02月04日
    浏览(57)
  • 【数据结构】实现单链表的增删查

    链表类和节点类的定义: 图解: 从中间位置插入: 图解:假定index=2 尾插: 删除当前线性表中索引为index的元素,返回删除的元素值: 图解: 删除当前线性表中第一个值为element的元素: 删除当前线性表中所有值为element的元素: 将当前线性表中index位置的元素替换为eleme

    2024年02月14日
    浏览(164)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包