题目
题目:给定单链表头节点,将单链表的链接顺序反转过来
例:
输入:1->2->3->4->5
输出:5->4->3->2->1
要求:按照两种方式实现
解决办法
方式一:(直接迭代法)
思路
单链表的结构如下所示:
如图所示。我们分析一下,如果要完成翻转需要思考什么:
- 循环遍历所有节点,直到结束。所以需要确认结束条件
- 安全地将当前节点指向它的前驱节点(直接指向的话会导致链表遍历失败,如下所示,找不到下一个操作节点了)
对于第1点,既然要循环遍历,那无非就是迭代器或者循环了,显然这里没办法使用迭代器,所以考虑使用while
循环。何时结束呢?不难发现,就是【当前节点的next节点为null的时候】,或者我们新增一个节点currNode
,表示当前操作节点,它随着遍历深度,不断往后移。显然它初始值是head
头节点,所以可以使用currNode == null
表示遍历结束了。
对于第2点,如何才能安全地将当前节点指向它的前驱节点呢?正常来说需要以下2步才能保证:
- 记录当前节点原有的next指向节点,这里记录为
oldNextNode
(原因见上图) - 还需要一个前驱节点
preNode
记录下一个节点需要设置的前驱节点(也可以看上图,除了找不到next节点以外,其实next节点也找不到前面的节点了,所以也需要记录)
准备好了我们就可以尝试模拟了。
操作步骤:
- 设置
oldNextNode
为currNode
的next节点 - 将
currNode
的next节点指向preNode
preNode
设置为currNode
currNode
设置为oldNextNode
似乎没问题,那我们开始写代码吧
代码示例
代码如下:
public class ReverseLinkedList {
public static void main(String[] args) {
ListNode node5 = new ListNode(5, null);
ListNode node4 = new ListNode(4, node5);
ListNode node3 = new ListNode(3, node4);
ListNode node2 = new ListNode(2, node3);
ListNode node1 = new ListNode(1, node2);
reverseLinkedList(node1);
}
private static void reverseLinkedList(ListNode head) {
// 初始化我们的中间节点
ListNode currNode = head;
ListNode oldNextNode = null;
ListNode preNode = null;
// 循环遍历,currNode != null 则表示还没遍历到结尾
while (currNode != null) {
oldNextNode = currNode.next;
currNode.next = preNode;
preNode = currNode;
currNode = oldNextNode;
}
System.out.println("试试看");
System.out.println(preNode);
}
}
最后我们看看结果吧:
倒过来了!
方式二:(递归法)
思路
(PS:既然说是递归解决了,那我们得先知道,递归能做什么。可以这么说,凡是能用递归解决的问题,一定是能将某个大问题,划分成N个存在共性的小问题。所以,我们可以朝这个方向看看。)
还是看着单链表的结构图来思考:
我们发现,如果从最后一个节点开始出发的话,这个题目的问题就会变成了另一个问题:我的前驱节点是谁?
在正常的遍历中,我们使用preNode
状态节点来记录前驱节点,但在递归里面就没那么麻烦了!因为在正向递归遍历节点的时候,当前节点的结束,肯定是在前驱节点的调用处。效果图如下:
递归代码如下所示:
public static void main(String[] args) {
ListNode node5 = new ListNode(5, null);
ListNode node4 = new ListNode(4, node5);
ListNode node3 = new ListNode(3, node4);
ListNode node2 = new ListNode(2, node3);
ListNode node1 = new ListNode(1, node2);
// reverseLinkedList(node1);
recursion(node1);
}
private static ListNode recursion(ListNode curr) {
// 递归出口(遍历到最后了)
if (curr == null || curr.next == null) {
return curr;
}
ListNode recursion = recursion(curr.next);
System.out.println(recursion.value);
return curr;
}
// 系统输出:
// 5
// 4
// 3
// 2
既然知道了这个特点,我们只需要做如下步骤就好了:
- 每次递归调用,返回当前节点(这是为了创造当前节点能跟前驱节点在同一个代码块里的环境)
- 获得递归调用返回的节点
recursion
之后,使用recursion .next = curr; curr.next = null;
即可
代码示例
public static void main(String[] args) {
ListNode node5 = new ListNode(5, null);
ListNode node4 = new ListNode(4, node5);
ListNode node3 = new ListNode(3, node4);
ListNode node2 = new ListNode(2, node3);
ListNode node1 = new ListNode(1, node2);
// reverseLinkedList(node1);
recursion(node1);
System.out.println(node5);
}
private static ListNode recursion(ListNode curr) {
// 递归出口(遍历到最后了)
if (curr == null || curr.next == null) {
return curr;
}
ListNode recursion = recursion(curr.next);
recursion.next = curr;
curr.next = null;
return curr;
}
最后我们看看结果吧: