前言
-
对于单链表的OJ练习,<font color=blue size=4>需要深刻理解做题的思路</font>,这样我们才能够在任何场景都能够熟练的解答有关链表的问题。
-
关于OJ练习(1):==->== 传送门 ==<-==,其题目较为简单,思路也好理解,本章与
(1)
差不多,难度不大,<font color=orange size=4>核心点就在于理解解题思路。</font>
链表的中间结点
题目链接:==->== 传送门 ==<-==。
- 该题目描述为:<font color=red size=4>给你单链表的头结点 head ,请你找出并返回链表的中间结点。如果有两个中间结点,则返回第二个中间结点。</font>
1.如果节点数为奇数,这个中间节点就显而易见了。(3)
2.如果节点数为偶数,这里认为中间两个节点的第二个节点为中间节点。(4)
<font color=red size=4>思路一</font>:
-
我们可以先遍历一遍链表,统计一下链表节点的个数。
-
然后将这个个数除以二加一(count / 2 + 1)便是中间这个节点的位置。
-
当然,我们在循环寻找这个中间节点的时候,是从头节点开始的,因此循环只需要循环
((count / 2 + 1) - 1)
即可。
<font color=blue size=4>代码实现:</font>
struct ListNode* middleNode(struct ListNode* head){
struct ListNode* cur = head;
int count = 0;
while (cur)
{
count ++ ;
cur = cur->next;
}
count = count / 2 + 1;
while (-- count) // 循环count - 1次
{
head = head->next;
}
return head;
}
<font color=red size=4>思路二</font>:
-
该做法为快慢指针。
-
啥为快慢指针呢?在本题有关当中,我们定义两个指针指向链表的头节点,并且共同遍历链表,不同的是,一个指针每一次走两步,另一个指针每次走一步,这就是快慢指针。
-
每当快指针满足循环结束条件,慢指针都是指向链表的中间节点的。因为快指针走两步,慢指针走一步,整个移动的位移差相差一倍,所以每当快指针满足结束条件的时候,慢指针走的步数都是快指针走的步数的一半, 因此慢指针指向的那个节点就是整个链表的中间节点。
-
快指针结束的条件有两种情况,一种是快指针刚好指向空结束,一种是快指针指向尾节点结束,也就是快指针的
next
为NULL
。
1.当快指针刚好指向NULL
结束,此时链表的节点个数为偶数:
2.当快指针指向尾节点结束,也就是快指针的next
为NULL
,此时链表的节点个数为奇数:
<font color=blue size=4>代码实现:</font>
struct ListNode* middleNode(struct ListNode* head){
struct ListNode* fast = head, * slow = head;
while (fast && fast->next)
{
fast = fast->next->next;
slow = slow->next;
}
return slow;
}
<font color=red size=4>注意:</font>
while
循环的判断条件,fast
一定要在前面,这是因为:判断是从左到右判断的,如果fast->next
在前,而此时链表的结点的个数为偶数,那么fast
就会直接到达NULL
,这时候对空指针解引用操作就出问题了。
链表中倒数第k个结点
题目链接:==->== 传送门 ==<-==。
- 该题目描述为:<font color=red size=4>输入一个链表,输出该链表中倒数第k个结点。。</font>
<font color=red size=4>思路一</font>:
-
既然是倒数第
k
个,那我们就看看是正数的第几个。 -
先遍历一遍单链表,统计一下链表的结点个数(
count
),通过数学知识,可得倒数的第k
个结点就是正数的第count - k + 1
个节点,这时只要在遍历一次链表,找到第count - k + 1
个节点返回即可。 -
当然,这里嘚注意
k
是不是大于链表节点的个数的情况。
<font color=blue size=4>代码实现:</font>
struct ListNode* FindKthToTail(struct ListNode* pListHead, int k ) {
struct ListNode* cur = pListHead;
int count = 0;
// 统计链表节点的个数
while (cur)
{
count ++ ;
cur = cur->next;
}
// 如果k大于链表的节点个数,直接返回NULL
if (k > count) return NULL;
int tmp = count - k;
// 由于从头个节点开始算,因此只需要循环count - k次就可以找到倒数第k个节点
while (tmp -- )
{
pListHead = pListHead->next;
}
return pListHead;
}
<font color=red size=4>思路二</font>:
- 同样是快慢指针,这里的快慢指针解法是:快指针先向后走
k
步或者先向后走k - 1
步,然后快指针与慢指针同时向后走,当快指针满足循环结束条件停止,此时慢指针指向的节点就是倒数第k
个节点。
1.如果快指针先向后走k
步,此时快指针与慢指针之间相差k
步,因此,当快指针到达NULL
时,此时慢指针刚好指向倒数第k
个节点。(倒数第k
个节点与NULL
相差k
步)(循环结束条件:fast == NULL
)
2.如果快指针先向后走k - 1
步,此时快指针与慢指针之间相差k - 1
步,然后快指针与慢指针同时向后走,当快指针满足循环结束条件停止,此时慢指针指向的节点就是倒数第k
个节点。(倒数第k
个节点与尾节点
相差k - 1
步)(循环结束条件:fast->next == NULL
)
<font color=blue size=4>代码实现:(这里只实现fast先走k步的情况,fast先走k - 1的情况大同小异)</font>
struct ListNode* FindKthToTail(struct ListNode* pListHead, int k ) {
struct ListNode* slow = pListHead, * fast = pListHead;
// fast先走k步
while (k -- )
{
if (!fast) return NULL; // 如果k还没为0但fast已经指向空了,说明k大于链表的结点的个数,此时直接返回NULL
fast = fast->next;
}
// 当fast为NULL时结束循环
while (fast)
{
fast = fast->next;
slow = slow->next;
}
return slow;
}
写在最后
<font color=red size=4>对于单链表的题目练习,最重要的是思路,我们在数据结构阶段要养成画图的习惯,因为它能帮助我们更好的理解。后续还会有单链表相关的题目练习。</font>
<font color=blue size=4>感谢阅读本小白的博客,错误的地方请严厉指出噢!</font>
标签:count,结点,OJ,fast,next,链表,节点,指针 From: https://blog.51cto.com/u_16019252/6474510