mirror of
https://github.com/youngyangyang04/leetcode-master.git
synced 2025-08-02 20:28:28 +08:00
120 lines
4.3 KiB
Markdown
120 lines
4.3 KiB
Markdown
<p align="center">
|
||
<a href="https://mp.weixin.qq.com/s/QVF6upVMSbgvZy8lHZS3CQ"><img src="https://img.shields.io/badge/知识星球-代码随想录-blue" alt=""></a>
|
||
<a href="https://mp.weixin.qq.com/s/b66DFkOp8OOxdZC_xLZxfw"><img src="https://img.shields.io/badge/刷题-微信群-green" alt=""></a>
|
||
<a href="https://img-blog.csdnimg.cn/20201210231711160.png"><img src="https://img.shields.io/badge/公众号-代码随想录-brightgreen" alt=""></a>
|
||
<a href="https://space.bilibili.com/525438321"><img src="https://img.shields.io/badge/B站-代码随想录-orange" alt=""></a>
|
||
</p>
|
||
<p align="center"><strong>欢迎大家参与本项目,贡献其他语言版本的代码,拥抱开源,让更多学习算法的小伙伴们收益!</strong></p>
|
||
|
||
|
||
> 反转链表的写法很简单,一些同学甚至可以背下来但过一阵就忘了该咋写,主要是因为没有理解真正的反转过程。
|
||
|
||
# 206.反转链表
|
||
|
||
https://leetcode-cn.com/problems/reverse-linked-list/
|
||
|
||
题意:反转一个单链表。
|
||
|
||
示例:
|
||
输入: 1->2->3->4->5->NULL
|
||
输出: 5->4->3->2->1->NULL
|
||
|
||
# 思路
|
||
|
||
如果再定义一个新的链表,实现链表元素的反转,其实这是对内存空间的浪费。
|
||
|
||
其实只需要改变链表的next指针的指向,直接将链表反转 ,而不用重新定义一个新的链表,如图所示:
|
||
|
||

|
||
|
||
之前链表的头节点是元素1, 反转之后头结点就是元素5 ,这里并没有添加或者删除节点,仅仅是改表next指针的方向。
|
||
|
||
那么接下来看一看是如何反转呢?
|
||
|
||
我们拿有示例中的链表来举例,如动画所示:
|
||
|
||

|
||
|
||
首先定义一个cur指针,指向头结点,再定义一个pre指针,初始化为null。
|
||
|
||
然后就要开始反转了,首先要把 cur->next 节点用tmp指针保存一下,也就是保存一下这个节点。
|
||
|
||
为什么要保存一下这个节点呢,因为接下来要改变 cur->next 的指向了,将cur->next 指向pre ,此时已经反转了第一个节点了。
|
||
|
||
接下来,就是循环走如下代码逻辑了,继续移动pre和cur指针。
|
||
|
||
最后,cur 指针已经指向了null,循环结束,链表也反转完毕了。 此时我们return pre指针就可以了,pre指针就指向了新的头结点。
|
||
|
||
# C++代码
|
||
|
||
## 双指针法
|
||
```C++
|
||
class Solution {
|
||
public:
|
||
ListNode* reverseList(ListNode* head) {
|
||
ListNode* temp; // 保存cur的下一个节点
|
||
ListNode* cur = head;
|
||
ListNode* pre = NULL;
|
||
while(cur) {
|
||
temp = cur->next; // 保存一下 cur的下一个节点,因为接下来要改变cur->next
|
||
cur->next = pre; // 翻转操作
|
||
// 更新pre 和 cur指针
|
||
pre = cur;
|
||
cur = temp;
|
||
}
|
||
return pre;
|
||
}
|
||
};
|
||
```
|
||
|
||
## 递归法
|
||
|
||
递归法相对抽象一些,但是其实和双指针法是一样的逻辑,同样是当cur为空的时候循环结束,不断将cur指向pre的过程。
|
||
|
||
关键是初始化的地方,可能有的同学会不理解, 可以看到双指针法中初始化 cur = head,pre = NULL,在递归法中可以从如下代码看出初始化的逻辑也是一样的,只不过写法变了。
|
||
|
||
具体可以看代码(已经详细注释),**双指针法写出来之后,理解如下递归写法就不难了,代码逻辑都是一样的。**
|
||
```C++
|
||
class Solution {
|
||
public:
|
||
ListNode* reverse(ListNode* pre,ListNode* cur){
|
||
if(cur == NULL) return pre;
|
||
ListNode* temp = cur->next;
|
||
cur->next = pre;
|
||
// 可以和双指针法的代码进行对比,如下递归的写法,其实就是做了这两步
|
||
// pre = cur;
|
||
// cur = temp;
|
||
return reverse(cur,temp);
|
||
}
|
||
ListNode* reverseList(ListNode* head) {
|
||
// 和双指针法初始化是一样的逻辑
|
||
// ListNode* cur = head;
|
||
// ListNode* pre = NULL;
|
||
return reverse(NULL, head);
|
||
}
|
||
|
||
};
|
||
```
|
||
|
||
|
||
|
||
## 其他语言版本
|
||
|
||
|
||
Java:
|
||
|
||
|
||
Python:
|
||
|
||
|
||
Go:
|
||
|
||
|
||
|
||
|
||
-----------------------
|
||
* 作者微信:[程序员Carl](https://mp.weixin.qq.com/s/b66DFkOp8OOxdZC_xLZxfw)
|
||
* B站视频:[代码随想录](https://space.bilibili.com/525438321)
|
||
* 知识星球:[代码随想录](https://mp.weixin.qq.com/s/QVF6upVMSbgvZy8lHZS3CQ)
|
||
<div align="center"><img src=../pics/公众号.png width=450 alt=> </img></div>
|