数据结构之链表深度讲解

小伙伴们,大家好呀,上次听我讲完顺序表想必收获不少吧,嘿嘿,这篇文章你也一样可以学到很多,系好安全带,咱们要发车了。

因为有了上一次顺序表的基础,所以这次我们直接进入正题,温馨提示:本篇文章的内容会比上次的顺序表文章的内容更加难,所以请耐心看完,过程中我会将一些难点,易混淆点给大家一一细讲,如有需要自行做笔记。


链表结构体内的元素

和顺序表里一样创建一个自定义类型的结构体,并且创建一个自定义类型的变量,以及指向下一个节点的指针next。

链表的尾插

链表和上次的顺序表一样也有尾插,那么我先教大家在链表头文件里如何写这个函数。也许会有小伙伴说函数不是很简单吗,而且看了博主这么多的文章,写个函数有何难呢。其实事情不是你想的那么简单,且听我细细道来。

首先,我们先来看一下两张图的区别

顺序表在测试文件中的尾插函数

链表在测试文件中的尾插函数

从图中我们可以看到俩图的区别

相同点:尾插时都是传了地址和要插入的数据

不同点:创建时,顺序表:结构体缩名   表名         链表:用指针的形式定义第一个节点   

那么这时我们也就可以在链表头文件内写尾插函数了,代码如下:

也许这时就会有人懵了,诶,我当时只是传了节点的地址呀,怎么这里就突然用了二级指针了?

所以说,话可不能说得太满。

之前我就已经详细的介绍了,我们在创建第一个节点时是以指针的形式创建,现在又传地址(也就是说传的是指针的地址)给尾插函数,所以这里就应该用二级指针啦。

那么基本上就给大家讲清楚啦,这是本篇文章的第一个难点

那么接下来我们来看看尾插代码主体部分具体怎么写吧。

思路:和顺序表一样先判断是否为空,因为我们创建好的第一个节点是空的,因此我们要去给它申请空间(这个函数具体怎么写放在下面,一会儿会简单讲一下),然后判断第一个节点是否为空,如果第一个节点为空,那么直接用“=”插入即可。温馨提醒:指针   =   指针的意思就是与变量与变量之间的赋值是一样的。那么当二级指针不为空时,我们需要以指针的形式创建尾节点,并且规定尾节点指向下一个的节点的地址不为空,也就是while(ptail->next),让ptail指针遍历链表,链表为空时插入数据。其中patil   =   ptail->next   的意思是ptail这个指针已经移到了下一个节点处, 那么   ptail->next   =    newnode  意思就是ptail指针连接到了newnode这个节点,但指针ptail还未移动到newnode处。为了方便大家理解,可以看一下下面的图

简单地给大家画了一下,差不多就这个意思,希望大家能理解。

那么尾插函数已经介绍完了,我们再来简单过一下链表中如何申请空间。

链表中申请节点空间

基本上和顺序表的差不太多,不一样的地方有顺序表里使用的是realloc(增容),而我们这里需要的是申请空间,因此我们可以用malloc,也可以使用calloc,大家有兴趣可以自己去尝试一下。申请完空间后,也许有人就不知道该干什么了,兄台别忘了我们是要把这个数据给尾插进去的呀,因此就得到了newnode->data =  x,插入完毕后也不要忘记将next指针置为空哦。

那么到此为止我们的链表尾插算是真正讲完了。尾插讲完后,我们就要开始起飞了,请各位小伙伴们系好安全带,飞机即将起飞。

链表的头插

链表头文件里的头插函数和上面图片里的一样,这里就不浪费文字去讲解了。毕竟我们主要还是将链表源文件的头插函数内部的写法和思路,首先我们来思考一个问题:假设我们就往一个空链表中进行插入数据,那么请问这个数据的插入方式究竟是头插还是尾插?想必大家的答案都是无法确定该数据的插入方式。因此我们可以从这里得到一个结论:头插数据时必须确保链表不为空,那么我们就可以确定这个函数上来就可以先给它断言:链表不为空。紧接着我们来看一下下面这张图

大家可以看到:这是一个不为空的链表,此时我们要头插数据,但是前面没有节点,因此我们需要创建节点,并且插入数据,因此也就有了代码图中的第二行代码,并且让newnode的next指针指向下一个节点,也就是我们在创建新节点之前的头节点pphead,并且此时pphead已经不再是头节点了,因此,我们需要用赋值符号“=”将pphead移到前面去(换句话说就是将两个指针互换了一下位置,但我不推荐这么理解)。

学习指南:看到这里的小伙伴如果感觉还吃得消,那么请继续看头插与尾插的拓展,如果有吃不消的小伙伴,可以先下滑看链表的尾删。

头插与尾插的拓展

指定位置的尾插

首先,我们需要确保所指定的位置不能为空,其次节点与节点之间没有空间,如图所示

因此我们需要使用malloc申请空间,创建新节点

创建完新的节点之后我们就要将其与其他节点连接起来,那么我们先连newnode的尾部吧,我们需要将newnode->next 指向原先pos所指向的下一个位置,也就是newnode->next   =    pos->next, newnode的前面就是与pos相连接即可,也就是让pos->next指向newnode

指定位置的头插

首先,我们需要确保链表不为空,并且头节点也不为空(为什么不能为空,一会给大家解释),对于我们来说最容易想到的是第二种情况

思路:我们需要定义一个新的指针使该指针从头节点phead开始,当prev的下一个节点是pos时,停下,并头插newnode这个节点。

第一种思路往往是我们容易忽略的一种

这种情况,我们只需要使用if语句,条件就是pos == phead即可,之后再调用一下我们指前写的头插函数将该数据插入就完成了,所以不明白*phead不能为空的小伙伴们恍然大悟了吗?

那么如何找到指定位置呢?

思路:我们先重新定义一个头节点pcur,通过循环找到与x相等的节点并返回即可

指定位置的删除

首先保证链表头节点和指定位置不为空,那么这里我们分两种情况讨论:

第一种:当指定位置就是头节点时,直接调用头删函数

第二种:也就是如下图这种情况时

先找到pos的前一个节点,让它的next指向pos的下一个节点,如图

最后就是释放掉pos并且置为空即可

指定位置的尾删

指定位置的尾删思路还是比较简单的,我这就给大家细细道来

思路:首先保证指定位置和指定位置的下一个节点不为空,为了便于大家可以清晰的分辨和理解,于是我定义了一个del指针让它代替pos->next,让pos->next = del->next,之后释放掉del并置为空。

链表的删除

 尾删:

思路:分两种情况讨论:

第一种:链表里只有一个节点时,直接释放掉*pphead,并且置为空即可,而判断它的条件:*pphead->next  ==  NULL;

第二种:连表里有多个节点时,先定义两个指针,使其中一个指针开始循环,另一个指针记录进行循环的指针的位置,当循环等指针的next指向空时,释放掉一开始循环的指针,并置为空,以及保存循环指针位置的指针的next也指向空

头删:

头删的思路比尾删的要简单许多,代码如下

保证链表和头节点不为空的情况下,先定义next指针,它负责接收*pphead的next指针指向的下一个节点,释放*pphead,最后将next赋给*pphead

链表的销毁

首先声明链表以及头节点的指针不为空,重新定义一个头节点指针prev,让prev进入循环,条件是prev不为空,进入循环后,重新定义一个指针next,它负责保存prev->next,然后释放掉prev, 再将next  的地址传给prev,跳出循环后将*pphead置为空,这样链表就销毁了。


当然这篇文章看下来,你可能还是会有些模糊,没事,我们不是还有总结环节嘛,下面就进入咱们的总结环节

总结:

1.    eg.prev->next   表示指针指向下一个节点,但指针还未移动

2.    eg.prev = prev->next   表示指针指向下一个节点,指针已经移动至下一个节点

3.    prev  =  pcur   表示pcur指针已经将它的地址和数值全传给了prev

4.    链表思路上的小妙招:去寻找插入以及删除节点时哪一个链条或哪几条链条被影响了,找到被影响的链条后,将其修改一下链接对象即可

5.关于pphead、 *pphead、**pphead还不清楚的同学可以参考一下下图


最后也祝大家在这个五一玩的开心

如果,你喜欢我的文章,不妨给我点个关注呗,如果有在这方面的问题也可以随时联系我哦

  • 22
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值