【LeetCode链表#8】翻转链表(双指针+递归)/K个一组翻转
翻转链表
题意:反转一个单链表。
示例: 输入: 1->2->3->4->5->NULL 输出: 5->4->3->2->1->NULL
思路分析
双指针法是本体的最基本的解法,由此还可以改写为递归解法
双指针法
我们需要定义两个指针,pre和cur
初始时,cur指向头节点,pre指向null(pre需要在cur之前,前面又没有节点,那只能是null咯。或者可以这么理解:翻转之后此时pre的位置会变成链表尾部,那链表尾部肯定指向null,所以需要将pre初始化为null)
让cur指向前一个节点
此时为了保存cur的下一个节点,好翻转之后让指针顺利移动到下一位置,我们需要先有一个临时节点temp
然后先把pre移动到当前cur的位置
再根据temp将cur移动到正确的下移位置
注意:这里先移动cur再移动pre的话会导致pre移动到的并不是之前cur的位置,因为cur的值已经先于pre改变(刻舟求剑懂不懂?)
之后就重复上述步骤即可
那么什么时候结束循环呢?当cur指到原链表的尾部(null)时便可结束,此时pre指向的节点为新的头节点
代码
明确步骤之后代码就很好写了
class Solution {
public ListNode reverseList(ListNode head) {
//双指针法
//初始化两个指针
ListNode cur = head;
ListNode pre = null;
//定义用于存放cur下一节点的临时节点temp
ListNode temp;
while(cur != null){//从旧的头节点开始遍历,到翻转前的链表尾部(也就是null)结束
temp = cur.next;
cur.next = pre;
//注意,cur翻转完之后一定先让pre按照当前cur的值移动到指定位置,再让cur更新为temp
//不然cur先动了pre就找不到之前cur的位置了
pre = cur;
cur = temp;
}
return pre;//翻转结束,此时pre指向的就是新的头节点
}
}
c++版
思路一致,代码对应着写就可以
class Solution {
public:
ListNode* reverseList(ListNode* head) {
//双指针法
ListNode* pre = nullptr;
ListNode* cur = head;
while(cur != nullptr){
//保存cur的下一节点
ListNode* temp = cur->next;
//交换
cur->next = pre;
//一定要先移动pre
pre = cur;
cur = temp;
}
return pre;//此时的pre为新链表的head
}
};
递归法
递归法主要是针对双指针法在代码层面上的一个优化,原理层面与双指针法一致
因此递归法的代码可以与双指针法的一一对应
代码
LeetCode上解题模板中,Solution类给了一个主方法reverseList
那么根据递归的写法,我们还要写一个reverse方法,让reverseList去调用它来翻转链表
class Solution {
public ListNode reverse(ListNode pre, ListNode cur){
}
public ListNode reverseList(ListNode head) {
reverse();
}
}
reverseList传入的参数是head,那么reverse需要的两个参数怎么传?
参考双指针法的初始化部分,reverse中的两个参数也要对应进行初始化
即:
class Solution {
public ListNode reverse(ListNode pre, ListNode cur){
}
public ListNode reverseList(ListNode head) {
reverse(null, head);//与双指针法对应
}
}
接下来按照双指针思路把reverse的功能完善即可,完整代码如下:
class Solution {
public ListNode reverse(ListNode cur, ListNode pre){
ListNode temp;
if(cur == null){
return pre;
}
temp = cur.next;//保存cur.next
cur.next = pre;//翻转操作
//接下来要让pre和cur整体向后移动一位,在递归里应该直接用return来实现,下面代码对应于
//pre = cur;
//cur = temp;
return reverse(temp, cur);//进入下一层递归
}
public ListNode reverseList(ListNode head) {
return reverse(head, null);//与双指针法对应
}
}
c++版
c++的解题模板如下:
class Solution {
public:
ListNode* reverseList(ListNode* head) {
}
};
要是用递归的话,要再写一个成员函数reverse,该函数的返回类型仍然是ListNode*
class Solution {
public:
ListNode* reverse(ListNode* pre, ListNode* cur) {
}
ListNode* reverseList(ListNode* head) {
}
};
代码如下
class Solution {
public:
ListNode* reverse(ListNode* cur, ListNode* pre) {
//这里就不好用while了因为要递归
if(cur == nullptr){
return pre;
}
ListNode* temp = cur->next;
cur->next = pre;
//接下来要用return实现以下代码
// pre = cur;
// cur = temp;
//从而达到递归
return reverse(temp, cur);//对着要实现的代码来填就行了
}
ListNode* reverseList(ListNode* head) {
//这里要实现双指针法中的
// ListNode* pre = nullptr;
// ListNode* cur = head;
//部分代码
return reverse(head,nullptr);//对应这填即可
}
};
二刷易错点
1、节点翻转环节的顺序
当使用临时节点保存cur的下一节点并将cur指向pre后,应该先将pre移动到cur处,再移动cur
要不然会乱
2、递归实现
递归版本完全可以根据双指针版本写出来,因此记住双指针版本的实现原理即可
需要注意,在递归实现中,不要使用while
K个一组翻转链表
https://leetcode.cn/problems/reverse-nodes-in-k-group/description/
给你链表的头节点 head
,每 k
个节点一组进行翻转,请你返回修改后的链表。
k
是一个正整数,它的值小于或等于链表的长度。如果节点总数不是 k
的整数倍,那么请将最后剩余的节点保持原有顺序。
你不能只是单纯的改变节点内部的值,而是需要实际进行节点交换。
思路
这里需要控制翻转区间,也就是每k个范围翻转一次
如果使用之前的迭代的方式,区间控制逻辑会变得很复杂,很容易出错(双层for加一堆变量),所以这里换用递归的方式来做
具体就是我们先定义一个指针post,指向head,然后遍历k步,此时就有了区间[head, post],该区间长度为k
然后我们还是定义pre和cur,将区间内的节点全部翻转
翻转结束后,cur、post以及翻转前的head节点会重叠到一起,此时我们需要继续翻转下一个k区间,也就是说我们要找head->next
因此,调用递归函数继续重复之前的操作:移动post确定区间,使用pre和cur翻转区间
最后,post为空,返回此时的pre即可。
代码
写法上遵守递归的三步走
class Solution {
public:
ListNode* reverseKGroup(ListNode* head, int k) {
ListNode* post = head; //用于标定翻转区间
for(int i = 0; i < k; ++i){
if(post == nullptr) return head;//终止条件
post = post->next;
}
ListNode* pre = nullptr;
ListNode* cur = head;
while(cur != post){//翻转区间内的节点
ListNode* tmp = cur->next;
cur->next = pre;
pre = cur;
cur = tmp;
}
//区间翻转完成,此时的head又被移动(交换)到了post处
//之后就调用递归翻转下一个k区间
//head不能动,所以输入递归函数的应该是cur
head->next = reverseKGroup(cur, k);
return pre;//与常规翻转一样,最后pre会指向链表头部,cur会指向空
}
};
ACM模式
#include <iostream>
using namespace std;
struct ListNode {
int val;
ListNode* next;
ListNode(int x) : val(x), next(NULL) {}
};
class Solution {
public:
ListNode* reverseKGroup(ListNode* head, int k) {
ListNode* post = head; //用于标定翻转区间
for (int i = 0; i < k; ++i) {
if (post == nullptr) return head;//终止条件
post = post->next;
}
ListNode* pre = nullptr;
ListNode* cur = head;
while (cur != post) {//翻转区间内的节点
ListNode* tmp = cur->next;
cur->next = pre;
pre = cur;
cur = tmp;
}
//区间翻转完成,此时的head又被移动(交换)到了post处
//之后就调用递归翻转下一个k区间
//head不能动,所以输入递归函数的应该是cur
head->next = reverseKGroup(cur, k);
return pre;//与常规翻转一样,最后pre会指向链表头部,cur会指向空
}
};
int main() {
ListNode* head = new ListNode(1);
head->next = new ListNode(2);
head->next->next = new ListNode(3);
head->next->next->next = new ListNode(4);
head->next->next->next->next = new ListNode(5);
int k = 2;
Solution solution;
ListNode* res = solution.reverseKGroup(head, k);
while (res) {
cout << res->val << ' ';
res = res->next;
}
return 0;
}
【LeetCode链表#8】翻转链表(双指针+递归)/K个一组翻转的更多相关文章
- leetcode 24. 两两交换链表中的节点 及 25. K 个一组翻转链表
24. 两两交换链表中的节点 问题描述 给定一个链表,两两交换其中相邻的节点,并返回交换后的链表. 你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换. 示例: 给定 1->2-> ...
- [LeetCode] 25. Reverse Nodes in k-Group 每k个一组翻转链表
Given a linked list, reverse the nodes of a linked list k at a time and return its modified list. k ...
- [LeetCode]25. Reverse Nodes in k-Group k个一组翻转链表
Given a linked list, reverse the nodes of a linked list k at a time and return its modified list. k ...
- Java实现 LeetCode 25 K个一组翻转链表
25. K 个一组翻转链表 给你一个链表,每 k 个节点一组进行翻转,请你返回翻转后的链表. k 是一个正整数,它的值小于或等于链表的长度. 如果节点总数不是 k 的整数倍,那么请将最后剩余的节点保持 ...
- LeetCode 25 —— K 个一组翻转链表
1. 题目 2. 解答 首先,利用快慢指针确定链表的总结点数. 偶数个结点时,结点个数等于 i * 2. 奇数个结点时,结点个数等于 i * 2 + 1. 然后将链表的每 K 个结点划分为一组.循环对 ...
- leetcode 25. K 个一组翻转链表
# coding:utf-8 __author__ = "sn" """ 25. K 个一组翻转链表 给你一个链表,每 k 个节点一组进行翻转,请你返 ...
- LeetCode 25. K 个一组翻转链表 | Python
25. K 个一组翻转链表 题目来源:https://leetcode-cn.com/problems/reverse-nodes-in-k-group 题目 给你一个链表,每 k 个节点一组进行翻转 ...
- [LintCode] Reverse Nodes in k-Group 每k个一组翻转链表
Given a linked list, reverse the nodes of a linked list k at a time and return its modified list. If ...
- k个一组翻转链表(java实现)
题目: 给出一个链表,每 k 个节点一组进行翻转,并返回翻转后的链表. k 是一个正整数,它的值小于或等于链表的长度.如果节点总数不是 k 的整数倍,那么将最后剩余节点保持原有顺序. 示例 : 给定这 ...
- LeetCoded第25题题解--K个一组翻转链表--java--链表
链表 单链表:链表中的每个元素实际上是一个单独的对象,而所有对象都通过每个元素的引用字段链接在一起. 双链表:与单链表不同的是,双链表的每个节点都含有两个引用字段. 链表优点 灵活分配内存空间 能在O ...
随机推荐
- [转帖]Cat导致内存不足原因分析
背景 线上几亿的数据在回刷的时候容器服务会出现OOM而重启,导致任务中断 内存泄露分析 jmap -histo pid 找出了有几十亿的java.lang.StackTraceElement对象,找不 ...
- [转帖]JVM 参数
https://www.cnblogs.com/xiaojiesir/p/15636100.html 我们可以在启动 Java 命令时指定不同的 JVM 参数,让 JVM 调整自己的运行状态和行为,内 ...
- Core 文件的简单学习
背景 最近公司内经常出现jvm进程宕机的情况. 宕机之后没有产生jvm的dump文件.比如xxx.hprof 但是产生了 core.$pid的文件. 曾经在aarch64架构上宕机时曾经想学习一下co ...
- 最简单的以CentOS为base images 安装 Nodejs等操作的方法
镜像内安装NodeJS的简单方法 公司内有产品需要安装nodejs以便进行相关操作,Linux和Windows时没有问题,但是如果是镜像的话可能会稍微复杂一点, 这里简单进行一下总结, 以便备忘. 1 ...
- 浅析RobotFramework工具的使用 | 京东物流技术团队
1 简介 最近几年越来越多的公司都开始进行自动化测试的设计和布局了,自动化,顾名思义就是把以人为驱动的测试行为转化为机器执行的一种过程,并经常用于回归测试中,市面上也存在很多开源的自动化测试的工具和理 ...
- 学习MySQL中DDL语句的修改字段与删除字段,删除表
连接本地mysql语句 mysql -hlocalhost -uroot -proot 显示表结构 语法:desc 表名 查看某一个表结构以及注释 语法:show create table 表名 sh ...
- go generate命令简介
最近在研究kratos的使用,发现在kratos run之前会先运行go generate ./...命令. 这个命令之前没怎么用过,所以决定学习下该命令的用法. go generate是Go语言中的 ...
- Jekyll安装
本文来自Jekyll官方文档. 简介 Jekyll是一个静态网站生成器,它可以把Markdown写就的文本根据指定的布局生成静态网站.你可以调整网站的外观.URL.页面展示数据等信息. 准备工作 Je ...
- 【SpringBoot】当AOP引发的异常与@RestControllerAdvice擦肩而过:异常处理的盲点揭秘
各位上午/下午/晚上好呀! 今天在写bug的时候发现一个这样的问题: AOP抛出的异常竟然没有被@RestControllerAdvice注解修饰的异常统一处理类处理. 有一个需求,对某些加了自定义注 ...
- Flask 框架:运用Ajax实现数据交互
使用Ajax技术网页应用能够快速地将增量更新呈现在用户界面上,而不需要重载刷新整个页面,这使得程序能够更快地回应用户的操作,如下笔记将简单介绍使用AJAX如何实现前后端数据通信. 前后端发送字符串 前 ...