【力扣】反转链表I和II(迭代和递归)
前言
有句话叫做:如果面试官跟你看顺眼的话,就给你出一道反转链表,否则就出一道 hard。
所以反转链表不能不会吧,要不面试官想要你都没有机会了。
206. 反转链表
class Solution {
public ListNode reverseList(ListNode head) {
}
}
迭代
迭代就是遍历链表,需要有个 cur
指针在链表上游走。
先思考一般情况,假设遍历到某一节点,应该做什么?
如上图所示,应该修改其 next
指针,指向前一个节点。由于单链表的性质,无法从当前节点获取前一个节点的指针,所以需要有个 pre
指针记录当前节点的前一个节点。把 cur
的指针修改后,就无法获取其后一个节点了,因此需要有个 next
指针保存 cur
的下一个节点。更新 pre
和 cur
指针,让它们后移一位即可。
class Solution {
public ListNode reverseList(ListNode head) {
// pre记录cur的前一个节点,cur是当前遍历到的节点
ListNode pre = null, cur = head;
// 遍历链表一般用while循环
while(cur != null){
// 保存cur的下一个节点
ListNode next = cur.next;
// 修改cur指针,指向它前一个节点
cur.next = pre;
// 更新pre、cur, 同时后移一位
pre = cur;
cur = next;
}
// 此时cur为null, pre指向最后一个节点
return pre;
}
}
递归
递归有两个性质:
- 边界条件
- 转化为子问题调用自身
先思考第二个问题,如何转化为子问题调用自身?
转化为子问题也就是缩小问题的规模,如果 head
指向的链表有 n
个节点,那么 head.next
指向的链表就有 n - 1
个节点,因此 head.next
就是一个子问题。为了更好地复用递归函,需要明确递归函数的定义,这里递归函数的定义是,输入一个单链表的头指针,返回该链表反转后的链表的头指针。
调用 head.next
的示意图如下:
根据递归函数的定义,它计算后返回值的情况如下所示:
接下来,就是要处理 head 和 reverseList(head.next)
的关系了,这个例子中就是 1 号节点和后面部分的关系。这里有两步,第一步,让 2 号节点指向 1 号节点;第二步,让 1 号节点指向 null
。第一步翻译成代码就是head.next.next = head
,第二步翻译成代码就是head.next = null
。
现在,第二个问题就处理完了,代码如下
class Solution {
public ListNode reverseList(ListNode head) {
// 1、边界条件
// if(){
// }
// 2、转化为子问题调用自身
ListNode last = reverseList(head.next);
// 修改head和后面部分指针的关系
head.next.next = head;
head.next = null;
return last;
}
}
对于递归函数,我觉得第二部分是核心,边界条件是特殊情况,可以先不考虑。自己去想可能会遗漏一些情况,这时如果允许执行测试用例,可以借助执行测试用例来帮助我们补全边界条件。
比如,这里我暂时不清楚边界条件,先执行测试用例,看看它会报什么错,然后根据它的错误提示来写边界条件。
直接执行上面的代码,会报这样的错
这样就知道 head 不能为空,加上条件
// 1、边界条件
if(head == null){
return head;
}
再执行代码,会报如下的错
这样就知道 head.next 不能为空,加上条件
// 1、边界条件
if(head == null || head.next == null){
return head;
}
完整代码
class Solution {
public ListNode reverseList(ListNode head) {
// 1、边界条件
if(head == null || head.next == null){
return head;
}
// 2、转化为子问题调用自身
ListNode last = reverseList(head.next);
// 修改head和后面部分指针的关系
head.next.next = head;
head.next = null;
return last;
}
}
92. 反转链表 II
迭代
希望复用反转单链表的代码,这样就需要把需要反转的部分单独抽取出来,反转好了之后再拼接回去。因此需要有 4 个指针,leftNode 和 rightNode 分别指向需要反转部分的首尾节点,pre 指针指向 leftNode 的前一个,cur 指针指向 rightNode 的后一个,pre 和 cur 的作用是为了保存断开链表和拼接链表的位置。4 个指针的位置关系如下图所示。
确定 4 个指针的位置并不复杂,循环指定步数即可。确定好 4 个指针的位置后,就是断开 pre 和 leftNode、rightNode 和 cur,调用之前的反转单链表将中间部分反转,最后再拼接回去即可。
完整代码
class Solution {
public ListNode reverseBetween(ListNode head, int left, int right) {
// 虚拟头节点
ListNode dummy = new ListNode(-1, head);
ListNode pre = dummy, cur = null;
ListNode leftNode = null, rightNode = head;
// 从虚拟头节点走left-1步,来到left前一个节点
for(int i = 0; i < left - 1; i++){
pre = pre.next;
}
// left节点在pre节点下一个
leftNode = pre.next;
// 从头节点走right-1步,来到right节点
for(int i = 0; i < right - 1; i++){
rightNode = rightNode.next;
}
// cur节点在right节点下一个
cur = rightNode.next;
// 断开指针
pre.next = null;
rightNode.next = null;
// 反转between
reverseList(leftNode);
// 拼接指针
pre.next = rightNode;
leftNode.next = cur;
return dummy.next;
}
// 反转单链表
private ListNode reverseList(ListNode head){
ListNode pre = null, cur = head;
while(cur != null){
ListNode next = cur.next;
cur.next = pre;
pre = cur;
cur = next;
}
return pre;
}
}
递归
首先需要实现反转前 n 个节点的函数 ListNode reverseN(ListNode head, int n)
。
比如要反转如下链表的前 3 个节点
那就是要反转以 head.next 开头的链表的前 2 个节点
如果reverseN
函数写的是正确的,那么按照它的定义,返回值如下
这里又涉及到 head 和 reverseN(head.next, n - 1)
指针关系的处理,和递归反转单链表是类似的,第一步是让 2 号节点指向 1 号节点,第二步略有不同,反转单链表是直接让1号节点指向 null,而这里应该让 1 号节点指向 4 号节点,也就是不需要反转的部分。因此,就需要一个指针记录 4 号节点的位置。
reverseN
代码如下
// 记录不用反转的第一个节点
ListNode suc = null;
private ListNode reverseN(ListNode head, int n){
// 1、边界条件
if(n == 1){
suc = head.next;
return head;
}
// 2、转化为子问题调用自身
ListNode last = reverseN(head.next, n - 1);
// 修改head和后面部分指针的关系
head.next.next = head;
head.next = suc;
return last;
}
接下来实现reverseBetween
当 left 等于 1 的时候,就转化成了reverseN
的问题。
对于 head,反转 left 和 right;对于 head.next,应该反转 left - 1 和 right - 1;对于head.next.next,应该反转 left - 2 和 right - 2…
所以,reverseBetween
函数代码如下
public ListNode reverseBetween(ListNode head, int left, int right) {
if(left == 1){
return reverseN(head, right);
}
head.next = reverseBetween(head.next, left - 1, right - 1);
return head;
}
完整代码
class Solution {
public ListNode reverseBetween(ListNode head, int left, int right) {
// 当left==1, 就转化成立reverseN问题
if(left == 1){
return reverseN(head, right);
}
head.next = reverseBetween(head.next, left - 1, right - 1);
return head;
}
// 记录不用反转的第一个节点
ListNode suc = null;
private ListNode reverseN(ListNode head, int n){
// 1、边界条件
if(n == 1){
suc = head.next;
return head;
}
// 2、转化为子问题调用自身
ListNode last = reverseN(head.next, n - 1);
// 修改head和后面部分指针的关系
head.next.next = head;
head.next = suc;
return last;
}
}
参考资料
【力扣】反转链表I和II(迭代和递归)的更多相关文章
- 力扣Leetcode 45. 跳跃游戏 II - 贪心思想
这题是 55.跳跃游戏的升级版 力扣Leetcode 55. 跳跃游戏 给定一个非负整数数组,你最初位于数组的第一个位置. 数组中的每个元素代表你在该位置可以跳跃的最大长度. 你的目标是使用最少的跳跃 ...
- 力扣 - 剑指 Offer 55 - II. 平衡二叉树
题目 剑指 Offer 55 - II. 平衡二叉树 思路1(后序遍历+剪枝) 这题是上一题剑指 Offer 55 - I. 二叉树的深度的进阶,逻辑代码和那个一样,也是后续遍历,获取两个子节点较大的 ...
- Leetcode力扣45题 跳跃游戏 II
原题目: 跳跃游戏 II 给定一个非负整数数组,你最初位于数组的第一个位置. 数组中的每个元素代表你在该位置可以跳跃的最大长度. 你的目标是使用最少的跳跃次数到达数组的最后一个位置. 示例: 输入: ...
- 力扣 - 445. 两数相加 II
目录 题目 思路 代码实现 题目 给你两个 非空 链表来代表两个非负整数.数字最高位位于链表开始位置.它们的每个节点只存储一位数字.将这两数相加会返回一个新的链表. 你可以假设除了数字 0 之外,这两 ...
- 力扣 - 剑指 Offer 53 - II. 0~n-1中缺失的数字
题目 剑指 Offer 53 - II. 0-n-1中缺失的数字 思路1 排序数组找数字使用二分法 通过题目,我们可以得到一个规律: 如果数组的索引值和该位置的值相等,说明还未缺失数字 一旦不相等了, ...
- 力扣 - 剑指 Offer 57 - II. 和为s的连续正数序列
题目 剑指 Offer 57 - II. 和为s的连续正数序列 思路1(双指针/滑动窗口) 所谓滑动窗口,就是需要我们从一个序列中找到某些连续的子序列,我们可以使用两个for循环来遍历查找,但是未免效 ...
- 力扣题解-面试题58 - II. 左旋转字符串
题目描述 字符串的左旋转操作是把字符串前面的若干个字符转移到字符串的尾部.请定义一个函数实现字符串左旋转操作的功能. 比如,输入字符串"abcdefg"和数字2,该函数将返回左旋转 ...
- 力扣119. 杨辉三角 II
原题 1 class Solution: 2 def getRow(self, rowIndex: int) -> List[int]: 3 ans = [1] 4 for i in range ...
- 【力扣】剑指 Offer II 092. 翻转字符
题目 解题思路 一个很暴力的想法,在满足单调递增的前提下,使每一位分别取 1 或 0,去看看哪个结果小. 递归函数定义int dp(StringBuilder sb, int ind, int pre ...
随机推荐
- el-cascader组件根据最后一级向上找到父级并设置默认值
vue + elementUI项目中,el-cascader级联选择器使用频率非常高,一些基本使用方法可以参考elementUI官方文档,本文主要研究当接口只返回最后一级id时,如何向上找出所有父级数 ...
- 6.-Django设计模式及模版层
一.MVC (java等其他语言) MVC代表Model-view-Contorller(模型-视图-控制器)模式 M模型层主要用于对数据库层的封装 V视图层用于向用户展示结果 C控制器用于处理请求. ...
- 乾象投资:基于JuiceFS 构建云上量化投研平台
背景 乾象投资 Metabit Trading 成立于2018年,是一家以人工智能为核心的科技型量化投资公司.核心成员毕业于 Stanford.CMU.清北等高校.目前,管理规模已突破 30 亿元人民 ...
- 题解 AT2361 [AGC012A] AtCoder Group Contest
\(\sf{Solution}\) 显然要用到贪心的思想. 既然最终的结果只与每组强度第二大选手有关,那就考虑如何让他的值尽可能大. 其实,从小到大排个序就能解决,越靠后的值越大,使得每组强度第二大选 ...
- CSS布局秘籍(2)-6脉神剑
HTML系列: 人人都懂的HTML基础知识-HTML教程(1) HTML元素大全(1) HTML元素大全(2)-表单 CSS系列: CSS基础知识筑基 常用CSS样式属性 CSS选择器大全48式 CS ...
- dafny : 微软推出的形式化验证语言
dafny是一种可验证的编程语言,由微软推出,现已经开源. dafny能够自我验证,可以在VS Code中进行开发,在编辑算法时,写好前置条件和后置条件,dafny验证器就能实时验证算法是否正确. 在 ...
- 一个jsqlparse+git做的小工具帮我节省时间摸鱼
背景 前些时间做了个小工具解决了团队内数据库脚本检验&多测试环境自动执行的问题,感觉挺有意思,在这跟大家分享一下. 工具诞生之前的流程是这样: 1.开发人员先在开发环境编写脚本&执行: ...
- Springboot2.6集成Email
Springboot集成Email 老版本 这时候的JavaMailSender是受到Spring的托管的,我们只需要配置参数就行了 !如何判断是否是被Springboot托管的:以下代码IDEA会自 ...
- 小米路由器局域网设备ping不通
问题 手机和电脑在同一个局域网内,都连接上小米路由器,我发现电脑部署的服务局域网设备都访问不到,甚至ping不到,排除了防火墙问题,最终发现是路由器一个设置导致的. 解决 将原来的混合加密,更换为强加 ...
- BFS和DFS学习笔记
1 算法介绍 1.1 BFS Breadth First Search(广度优先搜索),将相邻的节点一层层查找,找到最多的 以上图为例,首先确定一个根节点,然后依次在剩下的节点中找已找出的节点的相邻节 ...