LeetCode 热题 100_K 个一组翻转链表(31_25_困难_C++)(四指针法)
题目描述:给你链表的头节点 head ,每 k 个节点一组举行翻转,请你返回修改后的链表。
k 是一个正整数,它的值小于或即是链表的长度。假如节点总数不是 k 的整数倍,那么请将最后剩余的节点保持原有顺序。
你不能只是单纯的改变节点内部的值,而是必要实际举行节点交换。
输入输出样例:
示例 1:
https://i-blog.csdnimg.cn/direct/89a07e76cb3a4002a65e7ffcf2ecb1f2.png
输入:head = , k = 2
输出:
示例 2:
https://i-blog.csdnimg.cn/direct/7bb058065cac443196e02a66bef5a120.png
输入:head = , k = 3
输出:
提示:
链表中的节点数目为 n
1 <= k <= n <= 5000
0 <= Node.val <= 1000
题解:
解题思绪:
思绪一(四指针法):
1、通过题目分析,在每次翻转前必要举行个数的判断,若满意再将k个结点翻转,将翻转后的答案举行连接。
我们发现我们在举行翻转的时候必要保存k个结点的首和尾(kHead和kTail),并且还必要保存kHead之前的一个结点(ansTail)和kTail之后的一个结点(next_kHead),方便将翻转后的链表举行连接和剩余结点的处理,因此我们必要四个指针(kHead、kTail、ansTail、next_kHead)。
具体实现思绪请看下图:
https://i-blog.csdnimg.cn/direct/a456f55cca63480a8075bf2dc7682bb7.jpeg
代码实现
代码实现(思绪一(四指针法)):
//判断剩余长度是否>=k,不够则返回nullptr,够则返回k个长度链表的尾结点
ListNode *judgeLen_k(ListNode *kHead,int k){
while(k-1){
if(kHead==nullptr){
return nullptr;
}
kHead=kHead->next;
--k;
}
return kHead;
}
//翻转固定个数的链表,返回翻转后的头结点
ListNode *reverseList_k(ListNode *kHead,int k){
ListNode *pre=nullptr,*r=kHead,*tmp=kHead;
while(k){
r=r->next;
tmp->next=pre;
pre=tmp;
tmp=r;
--k;
}
return pre;
}
//K 个一组翻转链表
ListNode* reverseKGroup(ListNode* head, int k) {
ListNode *dummyHead=new ListNode(0);
//存储答案的尾结点
ListNode *ansTail=dummyHead;
//交换前,k个结点的头
ListNode *kHead=head;
//交换前,k个结点的末尾,不够k个则为nullptr
ListNode *kTail=judgeLen_k(kHead,k);
//保存下一个区间的头
ListNode *next_kHead=nullptr;
while(kTail!=nullptr){
//保存下一个区间的头
next_kHead=kTail->next;
//翻转k个结点
reverseList_k(kHead,k);
//将k个结点翻转后的链表,连接到答案列表
ansTail->next=kTail;
kHead->next=next_kHead;
//更新答案链表的尾结点
ansTail=kHead;
//更新交换前,k个结点的头
kHead=next_kHead;
//判断之后的结点是否够k个
kTail=judgeLen_k(next_kHead,k);
}
ListNode *ansHead=dummyHead->next;
delete dummyHead;
return ansHead;
}
代码调试
#include<iostream>#include<vector>using namespace std;struct ListNode{ int val; ListNode *next; ListNode():val(0),next(nullptr){} ListNode(int x):val(x),next(nullptr){} ListNode(int x,ListNode *next): val(x),next(next){}}; //尾插法创建单链表ListNode *createList(vector<int> arr){ ListNode *head=nullptr,*tail=nullptr; for(const auto &val:arr){ if(head==nullptr){ head=tail=new ListNode(val); }else{ tail->next=new ListNode(val); tail=tail->next; } } return head;} //判断剩余长度是否>=k,不够则返回nullptr,够则返回k个长度链表的尾结点
ListNode *judgeLen_k(ListNode *kHead,int k){
while(k-1){
if(kHead==nullptr){
return nullptr;
}
kHead=kHead->next;
--k;
}
return kHead;
}
//翻转固定个数的链表,返回翻转后的头结点
ListNode *reverseList_k(ListNode *kHead,int k){
ListNode *pre=nullptr,*r=kHead,*tmp=kHead;
while(k){
r=r->next;
tmp->next=pre;
pre=tmp;
tmp=r;
--k;
}
return pre;
}
//K 个一组翻转链表
ListNode* reverseKGroup(ListNode* head, int k) {
ListNode *dummyHead=new ListNode(0);
//存储答案的尾结点
ListNode *ansTail=dummyHead;
//交换前,k个结点的头
ListNode *kHead=head;
//交换前,k个结点的末尾,不够k个则为nullptr
ListNode *kTail=judgeLen_k(kHead,k);
//保存下一个区间的头
ListNode *next_kHead=nullptr;
while(kTail!=nullptr){
//保存下一个区间的头
next_kHead=kTail->next;
//翻转k个结点
reverseList_k(kHead,k);
//将k个结点翻转后的链表,连接到答案列表
ansTail->next=kTail;
kHead->next=next_kHead;
//更新答案链表的尾结点
ansTail=kHead;
//更新交换前,k个结点的头
kHead=next_kHead;
//判断之后的结点是否够k个
kTail=judgeLen_k(next_kHead,k);
}
ListNode *ansHead=dummyHead->next;
delete dummyHead;
return ansHead;
}
int main(){ vector<int> a={1,2,3,4,5}; int k=2; ListNode *head=createList(a); ListNode *test=reverseKGroup(head,k); while(test!=nullptr){ cout<<test->val<<"->"; test=test->next; } cout<<"null"; return 0;} 反转链表(23_206_简单_C++)(单链表_迭代_递归):有关反转链表的知识请点击此链接
LeetCode 热题 100_K 个一组翻转链表(31_25)原题链接
接待大家和我沟通交换(✿◠‿◠)
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页:
[1]