一、反转整个链表

问题:定义一个函数,输入一个链表的头节点,反转该链表并输出反转后链表的头节点。

示例:

输入: 1->2->3->4->5->NULL
输出: 5->4->3->2->1->NULL
//单链表的实现结构
public class ListNode {
int val;
ListNode next;
ListNode(int x) { val = x;}
}

反转链表利用迭代不难实现,如果使用递归则有些许难度。

首先来看源码实现:

ListNode reverse(ListNode head) {
if(head == null || head.next == null)
return head;
ListNode ret = reverse(head.next);
head.next.next = head;
head.next = null;
return ret;
}

是否看起来不知所云,而又被这如此简洁的代码所震撼?让我们一起探索一下其中的奥秘。

对于递归算法,最重要的是明确递归函数的定义。

我们的reverse函数的定义如下:

输入一个节点head,将以head为起点的链表反转,并返回反转之后的头节点。

明白了函数的定义后,在来看这个问题。比如我们想反转这个链表

那么输入reverse(head)后,会在ListNode ret = reverse(head.next);进行递归

不要跳进递归!(你的脑袋能压几个栈呀?)

根据reverse函数的定义,函数调用后会返回反转之后的头节点,我们用变量ret接收

现在再来看一下代码

head.next.next = head;

接下来:

head.next = null;
return ret;

再跳出这层递归就会得到:

神不神奇,这样整个链表就反转过来了!

递归代码就是这么简洁优雅,但要注意两个问题:

1、递归函数要有base case,不然就会一直递归,导致栈溢出

if (head == null || head.next == null) return head;

即链表为空或只有一个节点,直接返回

2、当链表递归反转后,新的头节点为ret,而head变成了最后一个节点,应该令链表的某尾指向null

head.next = null;

理解这两个问题之后,我们可以进一步深入研究链表反转的问题,接下来的问题其实均为在这个算法上的扩展。

二、反转链表前N个节点

接下来我们来看这个问题:

问题:反转链表前N个节点,并返回链表头节点

说明:1 <= N <= 链表长度

示例:

输入: 1->2->3->4->5->NULL, n = 4
输出: 4->3->2->1->5->NULL

解决思路和反转整个链表差不多,只需稍加修改

ListNode successor = null; // 后驱节点(第 n + 1 个节点)

ListNdoe reverseN(ListNode head, int n) {
if (n == 1) {
successor = head.next;
return head;
}
// 以 head.next 为起点,需要反转前 n - 1 个节点
ListNode ret = reverseN(head.next, n - 1);
head.next.next = head;
head.next = successor; // 将反转后的 head 与后面节点连接
return ret;
}

具体区别:

1、base case 变为n == 1, 同时需要记录后驱节点

2、之前把head.next 设置为null,因为整个链表反转后,head变为最后一个节点。

现在head节点在递归反转后不一定为最后一个节点,故应记录后驱successor(第 n + 1 个节点), 反转之后将head连接上。

OK,如果这个函数你也能看懂,就离实现反转一部分链表不远了。

三、反转链表的一部分

现在我们开始解决这个问题,给一个索引区间[m, n](索引从1开始),仅仅反转区间中的链表元素。

说明:1 <= m <= n <= 链表长度

示例:

输入: 1->2->3->4->5->NULL, m = 2, n = 4
输出: 1->4->3->2->5->NULL

猛一看很难想到思路。

试想一下,如果m == 1,就相当于反转链表的前 n 元素嘛,也就是我们刚才实现的功能:

ListNode reverseBetween(ListNode head, int m, int n) {
//base case
if (m == 1) {
return reverseN(head, n); // 相当于反转前 n 个元素
}
// ...
}

那如果m != 1 该怎么办?

如果把head的索引视为1,那么我们是想从第m个元素开始反转;

如果把head.next的索引视为1,那么我们是想从第m - 1 个元素开始反转;

如果把head.next.next的索引视为1,那么我们是想从第m - 2 个元素开始反转;

......

区别于迭代思想,这就是递归的思想,所以我们可以完成代码:

ListNode reverseBetween(ListNode head, int m, int n) {
// base case
if (m == 1) {
return reverseN(head, n);
}
// 递归前进到触发 base case (m == 1)
head.next = reverseBetween(head.next, m - 1, n - 1);
return head;
}

至此,我们终于干掉了大BOSS!

利用递归方法实现链表反转、前N个节点反转以及中间部分节点反转的更多相关文章

  1. 利用grunt-contrib-connect和grunt-connect-proxy搭建前后端分离的开发环境

    前后端分离这个词一点都不新鲜,完全的前后端分离在岗位协作方面,前端不写任何后台,后台不写任何页面,双方通过接口传递数据完成软件的各个功能实现.此种情况下,前后端的项目都独立开发和独立部署,在开发期间有 ...

  2. 利用 C++ 单向链表实现队列

    利用C++ 单向链表实现数据结构队列,其实和上一篇基本内容相同,仅仅是插入的时候在链表的尾部插入,取元素都是一样的,都从头部取. #pragma once #include "stdio.h ...

  3. 单链表的前K个的逆序输出

    单链表逆序输出也是常被面试官问到题算法题,所以自己就总结了一下,在此贴出算法,与小伙伴们相互交流. 首先要有三个指针,前两个分别指向首节点,首节点的下一个节点,第三个是临时指针,是为了储存首节点的下一 ...

  4. 在单链表的第i个位置后插入一个节点(阿里+腾讯等面试题总结)

    时间:2014.04.26 地点:基地 ------------------------- 一.题目 题目是非常easy和基础,就是在单链表的第i个位置后插入一个节点.要求写代码,5分钟之内完毕.面腾 ...

  5. Partition List(链表的插入和删除操作,找前驱节点)

    Given a linked list and a value x, partition it such that all nodes less than x come before nodes gr ...

  6. 通过数据库中的表,使用 MyEclipse2017的反向生成工具-->hibernate反转引擎引擎(MyEclipse2017自带的插件) 来反转生成实体类和对应的映射文件

    通过数据库中的表,使用 MyEclipse2017的反向生成工具-->hibernate反转引擎引擎(MyEclipse2017自带的插件) 来反转生成实体类和对应的映射文件   文章目录 Ja ...

  7. 利用BBED恢复UPDATE改动前的值

    转载请注明出处:http://blog.csdn.net/guoyjoe/article/details/30615151 实验步骤例如以下: 1.创建表guo_test1 gyj@PROD> ...

  8. java 中递归的实现 以及利用递归方法实现汉诺塔

    今天说下java语言中比较常见的一种方法,递归方法. 递归的定义 简单来说递归的方法就是"自己调用自己",通过递归方法往往可以将一个大问题简单化,最终压缩到一个易于处理的程度.对于 ...

  9. 利用快排partition求前N小的元素

    求前k小的数,一般人的想法就是先排序,然后再遍历,但是题目只是求前N小,没有必要完全排序,所以可以想到部分排序,而能够部分排序的排序算法我能想到的就是堆排序和快排了. 第一种思路,局部堆排序. 首先, ...

随机推荐

  1. JAVA的基本程序设计结构(下)

    字符串 Java没有内置的字符串类型,而是在标准Java类库中提供了一个预定义类,叫做 String. String e=""; //an empty String String ...

  2. Qt数据库 QSqlTableModel实例操作(转)

    本文介绍的是Qt数据库 QSqlTableModel实例操作,详细操作请先来看内容.与上篇内容衔接着,不顾本文也有关于上篇内容的链接. Qt数据库 QSqlTableModel实例操作是本文所介绍的内 ...

  3. 嵌入式linux下获取flash分区大小

    在嵌入式系统中,由于flash存储空间有限,或者是存储数据,实现数据的循环删除,需要获取到分区的使用情况,可以通过系统下的函数statfs来获取使用情况:实现代码如下: flashInfo.cpp # ...

  4. Django 环境下常用的模型设计

    Django 环境下常用的模型设计 用户表 继承 django.contrib.auth.model import AbstractUser AbstractUser 默认已经包含了很多字段了 id ...

  5. 数据结构C++实现邻接矩阵存储图

    定义邻接矩阵存储的图类.[实验要求] 1. 创建一个邻接矩阵存储的图: 2. 返回图中指定边的权值: 3. 查找图中某顶点的第一个邻接顶点.某顶点关于另一个顶点的下一个邻接顶点序号: 4. 图的深度优 ...

  6. nodejs版本DESede/CBC/PKCS5Padding算法封装(3des)

    最近对接了一个第三方支付项目,用的加密算法是根本没听过的:DESede/CBC/PKCS5Padding 这个算法真的是坑爹了,网上搜索了一堆只有java版本是正常的,nodejs版本的各种问题,我了 ...

  7. importTSV工具导入数据到hbase

    1.建立目标表test,确定好列族信息. create'test','info','address' 2.建立文件编写要导入的数据并上传到hdfs上 touch a.csv vi a.csv 数据内容 ...

  8. Jenkins配置总结

    1.配置全局 系统管理->全局工具配置 2.配置 自己安装安装jdk,git,以及maven 3.系统管理->系统配置 3.1配置Jenkins URL 3.2 配置SSH Servers ...

  9. 【Net】StreamWriter.Write 的一点注意事项

    背景 今天在维护一个旧项目的时候,看到一个方法把string 转换为 byte[] 用的是写入内存流的,然后ToArray(),因为平常都是用System.Text.Encoding.UTF8.Get ...

  10. idea git拉取、合并、处理冲突、提交代码具体操作

    早在两个月前我还在用eclipse开发,并且也发布的一些eclipse git的相关操作(操作都是本人亲自实践过的),但由于项目团队要求,开发工具统一用idea,实在不得已而为之切换了开发工具, 初次 ...