您现在的位置是:群英 > 开发技术 > 编程语言
C语言中链表的常见使用有哪些,怎样做
Admin发表于 2022-08-20 17:44:53670 次浏览
在实际案例的操作过程中,我们可能会遇到“C语言中链表的常见使用有哪些,怎样做”这样的问题,那么我们该如何处理和解决这样的情况呢?这篇小编就给大家总结了一些方法,具有一定的借鉴价值,希望对大家有所帮助,接下来就让小编带领大家一起了解看看吧。


我们今天将学习链表。以下是本篇文章正文内容,下面案例可供参考

一、链表的概念及结构

1.概念

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

(图片来自比特就业课)

上图是一个普通链表,它物理上是不连续的,那怎么让这些数据建立联系呢?链表每个位置会存放两个数据——1个是数据,另1个是指针

typedef int SLTDataType;//这里是方便将来改数据类型,在这里把int修改一下即可
typedef struct SlistNode//一个位置放多个数据用结构体
{
	int data;
	struct SlistNode*next;//指针是指向下一个节点,下一个节点也是结构体,所以这里是结构体指针

}SLTNode;//将结构体类型名用SLTNode重命名

我们用一张图来深入了解一下它的逻辑结构:

图示是链表的三个相连元素,第一个元素存放数据1和第二个元素的地址,然后以此类推。那到这里可能有同学会问:“那每个都是有存放下一个地址,最后一个节点存放谁的地址?”是这样的,最后一个节点存放的是NULL空指针,空指针也是作为链表结尾的标记
到这里,我们知道,链表的每个节点放的是当前节点内容和下一个节点的指针,那我们怎么找到这条链表呢?毕竟你第一个节点放的是第二个节点的指针,事实上,我们会定义一个指针指向第一个节点,以此来确定该链表

二、链表的使用

1.遍历整个链表

代码如下(示例):

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

我们以这张图来理解一下这段代码,我们创建一个结构体指针cur,并把链表首元素地址赋给cur,也就是说,cur指向了链表的首节点,我们知道首节点里是有第二个节点的地址的,我们通过cur = cur->next;可以找到第二段节点,以此类推。

2.尾插

(图片来自比特就业课)

想要尾插一个节点,我们首先要malloc(扩容函数)一块空间出来,开辟出那块空间之后,要把新节点与链表连接起来,就需要链表尾部节点的地址,我们用循环遍历得到,然后把新节点地址赋给之前链表的尾部节点即可
代码如下(示例):

void SListPushBack(SLTNode**pphead, SLTDataType x)
{
	SLTNode*newnode = (SLTNode*)malloc(sizeof(SLTNode));
	//开辟一块大小为一个SLTNode类型的空间出来,并强制转换成结构体指针赋给newnode
	newnode->data = x;
	newnode->next = NULL;
	if (*pphead == NULL)//防止一开始链表里面什么也没有,
	{//由于pphead是头节点的地址的地址,*解引用一下得到头结点的地址
		*pphead = newnode;
	}
	else
	{
		SLTNode*tail = *pphead;
		while (tail->next != NULL)//寻找尾节点的指针
		{
			tail = tail->next;
		}
		tail->next = newnode;//新增的节点地址赋给尾结点存储
	}
}

这里为什么用二级指针传参?解释如下:
我们进行尾插时,要先找到链表所在嘛,然后这个是靠链表头节点地址确定的,你传了一个地址过去,注意这个地址是实参,你实参过去函数里会再创建一个形参也是这个地址,然后进行操作,改变的是形参里的东西,实参不会受影响。这里也就是传值调用和传址调用的区别,为了形参可以影响到实参,我们用传址调用,也就是传地址的地址

3.头插

(图片来自比特就业课)

我们假设现在要在一个链表前面插入0这个数据(如上图所示),0所在地址为0x0006708,那是不是要把原链表首节点地址放到0这个节点里面,然后头节点地址更新为0这个新节点的地址。如下图:

ps:plist/phead表示链表首节点地址
代码如下(示例):

void SListPushFront(SLTNode**pphead, SLTDataType x)
{
	SLTNode*newnode = (SLTNode*)malloc(sizeof(SLTNode));
	//开辟一块大小为一个SLTNode类型的空间出来,并强制转换成结构体指针赋给newnode
	newnode->data = x;//插入节点的数据
	newnode->next = *pphead;//把原先节点首元素地址赋给新节点
	*pphead = newnode;//新节点首元素地址用新插入的节点地址赋值
}

4.头删

关于头删,很多同学可能有一种想法,就是直接把头结点free(对动态分配的内存进行释放)掉,剩下的就是我们需要的链表。但这里会有一个问题,就是你把头节点去了,因为链表是用地址连接的,我们原本是用头节点地址找到该链表,现在头节点去掉了,我们怎么知道剩下的链表在哪里。所以这里的解决方案是,先把第二个节点地址保存起来然后再释放掉头节点
代码如下(示例):

void SListPopFront(SLTNode**pphead)
{
	SLTNode*next = (*pphead)->next;
	free(*pphead);
	*pphead = next;
}

5.尾删

尾删和头删有同样的问题,我们能不能直接把尾部节点给free掉?答案也是不可以的,同学你细想一下,尾部节点是不是连接着倒数第二个节点,而倒数第二个节点保存着尾结点地址,你把尾结点free掉了,那倒数第二个节点保存的就是野指针了,这显然是不行的,所以我们需要把倒数第二个节点存储的指针变成空指针,然后free掉尾结点

void SListPopBack(SLTNode**pphead)
{
	if (*pphead == NULL)//如果节点本来就没有,那就没有删除的说法了
	{
		return;
	}
	else if ((*pphead)->next == NULL)//如果本来链表只有一个节点,你删除一个,之前会有一个指针指向首节点来记录这个链表,你现在把唯一的节点删除了,那个记录链表的指针就成野指针了
	{
		free(*pphead);
		*pphead = NULL;
	}
	else
	{
		SLTNode*prev = NULL;//prev为tail前个节点指针
		SLTNode*tail = *pphead;
		while (tail->next != NULL)
		{
			prev = tail;
			tail = tail->next;
		}
		free(tail);
		prev->next = NULL;
	}
}

这里还是有注意的点的,比如你要删除的链表本来就是空链表,删除就无从谈起了,还有就是原先链表只有一个节点,你删除一个,之前会有一个指针指向首节点来记录这个链表,你现在把唯一的节点删除了,那个记录链表的指针就成野指针了

6.任意位置插入数据

上图红色是原有链表,我们要在2和3直接插入一个30怎么做?首先我们要把2、30、3这3个节点连接起来,也就是说,2节点要指向30这个节点,30这个节点要指向3这个节点。这里如何操作呢?我们需要设计一个函数先找到2节点和3节点的地址,然后进行插入操作
查找函数和插入函数如下

SLTNode* SListFind(SLTNode*phead, SLTDataType x)//查找链表
{
	SLTNode*cur = phead;
	while (cur)//非空指针则进行循环
	{
		if (cur->data == x)//给定链表中一个数据进行查找
		{
			return cur;//返回所查找位置指针
		}
		cur = cur->next;
	}
	return NULL;
}
void SListInsert(SLTNode**pphead,SLTNode*pos, SLTDataType x)//在pos前插入x,x是要插入的数
{
	if (pos == *pphead)//原链表只有1个节点
	{
		SListPushFront(pphead, x);
	}
	else
	{
		SLTNode*newnode = (SLTNode*)malloc(sizeof(SLTNode));
		//开辟一块大小为一个SLTNode类型的空间出来,并强制转换成结构体指针赋给newnode
		newnode->data = x;
		newnode->next = NULL;//开辟1个新节点
		SLTNode*prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		prev->next = newnode;
		newnode->next = pos;//把新节点连接上去
	}
}

两个函数一起调用是这样的

if (pos)//防止该位置是空指针
	{
		SlistInsert(&plist, pos, 30);
	}
	SListPrint(&plist);
}

7.任意位置删除数据

比如说我们现在要把3删除,那这里就会出现一个需要:3删除了,要把2和4连接起来,然后把3节点释放掉

void SListErase(SLTNode**pphead, SLTNode*pos)//删除pos位置的值
{
	SLTNode*prev = *pphead;
	while (prev->next != pos)//找到3节点的位置
	{
		prev = prev->next;
	}
	prev->next = pos->next;//prev是2,pos是3,pos->next是4,要把4接上2
	free(pos);//2和4接上之后就可以free掉3了
}

ps:上述prev是2,pos是3这种是方便同学你理解,并不特指它真的是2和3

然后这里进行函数调用的时候,依然需要进行find定位一下需要删除位置地址

SLTNode*pos = SListFind(plist, 3);
	if (pos)//防止该位置是空指针
	{
		SListErase(&plist, pos);
	}

后记


以上就是关于“C语言中链表的常见使用有哪些,怎样做”的介绍了,感谢各位的阅读,希望文本对大家有所帮助。如果想要了解更多知识,欢迎关注群英网络,小编每天都会为大家更新不同的知识。

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:mmqy2019@163.com进行举报,并提供相关证据,查实之后,将立刻删除涉嫌侵权内容。

标签: 链表的使用
相关信息推荐
2022-09-06 17:50:20 
摘要:本篇文章给大家带来了关于JavaScript的相关知识,其中主要介绍了关于BOM浏览器对象模型的相关问题,包括了windows对象的常见事件、定时器、js执行机制等等内容,下面一起来看一下,希望对大家有帮助。
2022-05-17 11:45:47 
摘要:html5教程:本文为大家分享了html5实现调用摄像头并拍照功能的代码,具有一定的参考价值,希望能够帮助到大家。
2022-04-29 12:03:04 
摘要:给大家带来一篇关于JavaScript实现单击网页任意位置打开关闭窗口的代码的相关教程文章,内容涉及到JavaScript、单击、网页、任意位置、打开新窗口、关闭窗口、JavaScript实现单击网页任意位置打开新窗口与关闭窗口的方法等相关内容,更多关于JavaScript实现单击网页任意位置打开新窗口与关闭窗口的方法的内容希望能够帮助到大家。
云活动
推荐内容
热门关键词
热门信息
群英网络助力开启安全的云计算之旅
立即注册,领取新人大礼包
  • 联系我们
  • 24小时售后:4006784567
  • 24小时TEL :0668-2555666
  • 售前咨询TEL:400-678-4567

  • 官方微信

    官方微信
Copyright  ©  QY  Network  Company  Ltd. All  Rights  Reserved. 2003-2019  群英网络  版权所有   茂名市群英网络有限公司
增值电信经营许可证 : B1.B2-20140078   粤ICP备09006778号
免费拨打  400-678-4567
免费拨打  400-678-4567 免费拨打 400-678-4567 或 0668-2555555
微信公众号
返回顶部
返回顶部 返回顶部