单链表倒置。
之前在 第 2 题 大数相加的时候已经分享过了,这里直接贴过来。
首先看一下原链表。
总共需要添加两个指针,pre
和 next
。
初始化 pre
指向 NULL
。
然后就是迭代的步骤,总共四步,顺序一步都不能错。
一次迭代就完成了,如果再进行一次迭代就变成下边的样子。
可以看到整个过程无非是把旧链表的 head
取下来,添加的新链表头部。代码怎么写呢?
next = head -> next; //保存 head 的 next , 以防取下 head 后丢失
head -> next = pre; //将 head 从原链表取下来,添加到新链表上
pre = head;// pre 右移
head = next; // head 右移
接下来就是停止条件了,我们再进行一次循环。
可以发现当 head
或者 next
指向 null
的时候,我们就可以停止了。此时将 pre
返回,便是逆序了的链表了。
public ListNode reverseList(ListNode head) {
if (head == null)
return null;
ListNode pre = null;
ListNode next;
while (head != null) {
next = head.next;
head.next = pre;
pre = head;
head = next;
}
return pre;
}
-
首先假设我们实现了将单链表逆序的函数,
ListNode reverseListRecursion(ListNode head)
,传入链表头,返回逆序后的链表头。 -
接着我们确定如何把问题一步一步的化小,我们可以这样想。
把
head
结点拿出来,剩下的部分我们调用函数reverseListRecursion
,这样剩下的部分就逆序了,接着我们把head
结点放到新链表的尾部就可以了。这就是整个递归的思想了。 -
找到递归出口
当然就是如果结点的个数是一个,那么逆序的话还是它本身,直接 return 就够了。怎么判断结点个数是不是一个呢?它的
next
等于null
就说明是一个了。但如果传进来的本身就是null
,那么直接找它的next
会报错,所以先判断传进来的是不是null
,如果是,也是直接返回就可以了。
public ListNode reverseList(ListNode head) {
ListNode newHead;
if (head == null || head.next == null) {
return head;
}
newHead = reverseList(head.next); // head.next 作为剩余部分的头指针
// head.next 代表新链表的尾,将它的 next 置为 head,就是将 head 加到末尾了。
head.next.next = head;
head.next = null;
return newHead;
}
关于链表的题,记住更改指向的时候要保存之前的节点,不然会丢失节点。