小伙伴们,大家好呀,上次听我讲完顺序表想必收获不少吧,嘿嘿,这篇文章你也一样可以学到很多,系好安全带,咱们要发车了。
因为有了上一次顺序表的基础,所以这次我们直接进入正题,温馨提示:本篇文章的内容会比上次的顺序表文章的内容更加难,所以请耐心看完,过程中我会将一些难点,易混淆点给大家一一细讲,如有需要自行做笔记。
链表结构体内的元素
和顺序表里一样创建一个自定义类型的结构体,并且创建一个自定义类型的变量,以及指向下一个节点的指针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还不清楚的同学可以参考一下下图
最后也祝大家在这个五一玩的开心
如果,你喜欢我的文章,不妨给我点个关注呗,如果有在这方面的问题也可以随时联系我哦