牛客题解 | 反转链表_1

news/2025/2/25 19:24:14/文章来源:https://www.cnblogs.com/wc529065/p/18737059

题目

题目链接

题目的主要信息:
  • 给定一个长度为\(n\)的链表,反转该链表,输出表头
举一反三:

学习完本题的思路你可以解决如下题目:

JZ6. 从尾到头打印链表

方法一:迭代(推荐使用)

思路:

将链表反转,就是将每个表元的指针从向后变成向前,那我们可以遍历原始链表,将遇到的节点一一指针逆向即可。指针怎么逆向?不过就是断掉当前节点向后的指针,改为向前罢了。

cur.next = pre

具体做法:

  • step 1:优先处理空链表,空链表不需要反转。
  • step 2:我们可以设置两个指针,一个当前节点的指针,一个上一个节点的指针(初始为空)。
  • step 3:遍历整个链表,每到一个节点,断开当前节点与后面节点的指针,并用临时变量记录后一个节点,然后当前节点指向上一个节点,即可以将指针逆向。
  • step 4:再轮换当前指针与上一个指针,让它们进入下一个节点及下一个节点的前序节点。

图示:
alt

Java实现代码:

public class Solution {public ListNode ReverseList(ListNode head) {//处理空链表if(head == null) return null;ListNode cur = head;ListNode pre = null;while(cur != null){//断开链表,要记录后续一个ListNode temp = cur.next; //当前的next指向前一个cur.next = pre; //前一个更新为当前pre = cur; //当前更新为刚刚记录的后一个cur = temp; }return pre;}
}

C++实现代码:

class Solution {
public:ListNode* ReverseList(ListNode* pHead) {//处理空链表if(pHead == NULL)return NULL;ListNode* cur = pHead;ListNode* pre = NULL;while(cur != NULL){//断开链表,要记录后续一个ListNode* temp = cur->next; //当前的next指向前一个cur->next = pre; //前一个更新为当前pre = cur; //当前更新为刚刚记录的后一个cur = temp; }return pre;}
};

Python实现代码:

class Solution:def ReverseList(self , head: ListNode) -> ListNode:#处理空链表if not head:return Nonecur = headpre = Nonewhile cur:#断开链表,要记录后续一个temp = cur.next #当前的next指向前一个cur.next = pre#前一个更新为当前pre = cur #当前更新为刚刚记录的后一个cur = tempreturn pre

复杂度分析:

  • 时间复杂度:\(O(n)\),遍历链表一次,一次循环
  • 空间复杂度:\(O(1)\),常数级变量,无额外辅助空间使用
方法二:递归(扩展思路)

思路:

从上述方法一,我们可以看到每当我们反转链表的一个节点以后,要遍历进入下一个节点进入反转,相当于对后续的子链表进行反转,这可以看成是一个子问题,因此我们也可以使用递归,其三段式模版为:

  • 终止条件: 当到达链表尾,要么当前指针是空,要么下一个指针是空,就返回。
  • 返回值: 每一级返回反转后的子问题的头节点。
  • 本级任务: 先进入后一个节点作为子问题。等到子问题都反转完成,再将本级节点与后一个的指针反转。

具体做法:

  • step 1:对于每个节点我们递归向下遍历到最后的尾节点。
  • step 2:然后往上依次逆转两个节点。
  • step 3:将逆转后的本层节点,即反转后这后半段子链表的尾,指向null,返回最底层上来的头部节点。

Java实现代码:

public class Solution {public ListNode ReverseList(ListNode head) {//递归结束条件if(head == null || head.next == null)return head;//反转下一个ListNode newHead = ReverseList(head.next); //逆转本级节点head.next.next = head; //尾部设置空节点head.next = null; return newHead;}
}

C++实现代码:

class Solution {
public:ListNode* ReverseList(ListNode* pHead) {//递归结束条件if(pHead == NULL || pHead->next == NULL)return pHead;//反转下一个ListNode* newHead = ReverseList(pHead->next);//逆转本级节点pHead->next->next = pHead; //尾部设置空节点pHead->next = NULL; return newHead;}
};

Python实现代码:

import sys
#设置递归深度
sys.setrecursionlimit(100000) 
class Solution:def ReverseList(self , head: ListNode) -> ListNode:#递归结束条件if head is None or head.next is None:return head#反转下一个newHead = self.ReverseList(head.next)#逆转本级节点head.next.next = head #尾部设置空节点head.next = Nonereturn newHead

复杂度分析:

  • 时间复杂度:\(O(n)\),相当于递归遍历一次链表
  • 空间复杂度:\(O(n)\),递归栈深度为链表长度\(n\)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/889707.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

牛客题解 | 判断链表中是否有环

牛客题库题解题目 题目链接 题目主要信息:给定一个链表的头节点,判断这个链表是否有环 环形链表如下所示:举一反三: 学习完本题的思路你可以解决如下题目: BM4.合并有序链表 BM5.合并k个已排序的链表 BM7.链表中环的入口节点 BM8.链表中倒数最后k个节点 BM9.删除链表的倒数…

牛客题解 | 剪绳子

牛客题库题解题目 题目链接 题目的主要信息:把一根长度为\(n\)的绳子分成\(m\)段,每段长度都是整数 求每段长度乘积的最大值举一反三: 学习完本题的思路你可以解决如下题目: JZ83. 剪绳子(进阶版) JZ71. 跳台阶扩展问题 JZ42. 连续子数组的最大和 方法一:动态规划(推荐…

牛客题解 | 剪绳子(进阶版)

牛客题库题解题目 题目链接 题目的主要信息:把一根长度为\(n\)的绳子分成\(m\)段,每段长度都是整数 求每段长度乘积的最大值 由于答案过大,请对 998244353 取模举一反三: 学习完本题的思路你可以解决如下题目: JZ14. 剪绳子 方法:快速幂+快速乘法(推荐使用) 知识点1:贪心…

牛客题解 | 判断一个链表是否为回文结构

牛客题库题解题目 题目链接 题目的主要信息:给定一个链表的头节点,判读该链表是否为回文结构 回文结构即正序遍历与逆序遍历结果都是一样的,类似123321 空链表默认为回文结构举一反三: 学习完本题的思路你可以解决如下题目: BM4.合并有序链表 BM5.合并k个已排序的链表 BM6…

牛客题解 | 判断是不是二叉搜索树

牛客题库题解题目 题目链接 题目主要信息:判断给定的一棵二叉树是否是二叉搜索树 二叉搜索树每个左子树元素小于根节点,每个右子树元素大于根节点,中序遍历为递增序举一反三: 学习完本题的思路你可以解决如下题目: BM30. 二叉搜索树与双向链表 BM37. 二叉搜索树的最近公共…

牛客题解 | 判断是不是完全二叉树

牛客题库题解题目 题目链接 题目主要信息:判断给定二叉树是否为完全二叉树 首先我们需要知道什么是完全二叉树:叶子节点只能出现在最下层和次下层,且最下层的叶子节点集中在树的左部。 需要注意的是,满二叉树肯定是完全二叉树,而完全二叉树不一定是满二叉树。举一反三: 学…

linux下安装 elasticsearch

一、基础环境 操作系统环境:Red Hat Enterprise Linux Server release 6.4 (Santiago) ES版本:elasticsearch-7.8.0-linux-x86_64.tar.gz Jdk:Java(TM) SE Runtime Environment (build 1.8.0_144-b01) 二、安装 1、上传安装包到/opt目录下 2、解压cd /opt # tar -zxvf elas…

mysql 啥样的索引能提高查询性能呢?

前言 在前面几章中,我们知道了页里面是如何存储的,页又是如何编排的。 这样我们知道了,如何定位到页,如何定位到行了,这些对我们索引的了解非常有帮助的。 知道这些后,那么我们如何利用索引查询呢? 也就是说我们如何利用这种数据结构呢? 是不是全部的查询都能通过索引去…

06 常用损失函数介绍

在前文中我们使用的损失函数都是均方误差(MSE,Mean Squared Error),本篇介绍一些其他的损失函数形式,以及他们的不同用途。 1. 回归任务常用损失函数 1.1 均方误差(MSE, Mean Squared Error) 均方误差(MSE)是回归任务中最常用的损失函数之一,用于衡量模型预测值与真实…

啦啦啦啦啦啦啦啦啦

啦啦啦 啦啦啦啦啦啦啦啦啦 ABC221G 神秘题,将坐标轴转 \(45\),然后 bitset 优化背包,记录路径把刚刚被更新的找出来,然后 _Find_next,每个点只会记一次。 AGC050a 神秘题,想到 \(\log\),然后发现一下位置 \(x\) 走十次能到的区间是 \([1024x,1024x+1023]\),区间长度够…

破解 vLLM + DeepSeek 规模化部署的“不可能三角”

通过 FC GPU 预留实例的闲置计费功能,企业用户能在充分利用 vLLM 的强大功能的同时找到成本、性能、稳定的最佳平衡点,并保持开发和运维的高效性。无论是将 FC vLLM 函数直接对外提供服务,还是深度集成到现有系统中,或是通过 CAP 还是魔搭来简化部署,都能找到满足您业务需…

条形码编码规则全解析:从黑白条纹到数字世界的转换密码

条形码的编码规则是将字符(数字、字母等)转换为特定黑白条纹或矩阵结构的标准化方法,核心目的是让机器能够快速、准确地识别和解析信息。以下是常见条形码编码规则的简介: 一维条形码编码规则 1. ​基本原理通过不同宽度的黑白条纹​(或空格)组合表示字符。 每个字符对应…