【LeetCode_206】反转链表

【LeetCode_206】反转链表

刷爆LeetCode系列

LeetCode第206题:反转链表

github地址

有梦想的电信狗

前言

本文用C++实现LeetCode206题:反转链表


题目描述

在这里插入图片描述
在这里插入图片描述

题目与思路分析

目标分析

  1. 有单链表的头节点 head ,反转原链表
  2. 返回反转后的链表的头指针
  3. 提高要求:时间复杂度为O(n),空间复杂度为O(1)

思路一:反转链表的指针指向

思路:遍历一遍链表,将当前结点的next指针,指向其前驱节点,即可实现链表的反转。最终返回链表的最后一个节点即可(原链表的最后一个结点作为新链表的头结点

操作

  • head == nullptr时,为空链表,直接return nullptr;
  • 遍历链表
    • curNode:从头结点head开始,依次遍历链表,更改指针的指向
      • 当前结点的next指针,指向其前驱节点,因此需要提前保存当前结点的前驱节点
    • curPrev:记录curNode的前驱节点,方便反转指针指向,初始值为nullptr
    • curNext:提前保存curNode的后继结点,防止反转指针指向curNode无法移动到下一个结点
  • 更改指向
    • 保存curNode的下一个结点ListNode* curNext = curNode->next
    • 更改当前结点的next指针curNode->next = curPrev
    • curPrev和curNode依次向后移动
      • curPrev = curNode;
      • curNode = curNext;
  • 最终return curPrevcurPrev最后的位置就是链表的尾结点
在 while 循环内保存 curNext,保证了 curNode 一定不为空,避免了对空指针解引用可以保证一定可以取到 curNext,为空或非空
在这里插入图片描述
  • 链表只有一个结点的情况
在这里插入图片描述

思路二:取链表的结点,头插到新链表中

思路:创建一个新链表,头结点为newHead,初始为nullptr。遍历原链表,将原链表中的节点依次头插到新链表中,头插后更新newHead,即可实现链表的反转,最终返回newHead

操作

  • 遍历链表
    • curNode:从头结点head开始,依次遍历链表,进行头插操作。将当前结点的next指针,指向新链表的newHead,再更新newHead的值。之后curNode移向下一个结点。由于头插后curNode->next的值更改,不能通过curNode = curNode->next的方式移动,因此需要提前保存curNode的后继节点
    • curNext:提前保存curNode的后继结点,防止头插curNode无法移动到下一个结点
  • 头插
    • 保存curNode的下一个结点ListNode* curNext = curNode->next
    • 更改当前结点的next指针curNode->next = newHead
    • 更新newHead的值newHead = curNode;
    • curNode向后移动curNode = curNext;
  • 最终return newHeadnewHead即为新链表的头结点
在 while 循环内保存 curNext,保证了 curNode 一定不为空,避免了对空指针解引用可以保证一定可以取到 curNext,为空或非空
在这里插入图片描述

代码实现

思路一:反转指针指向

以下两种写法是保存curNext指针的方式不同

  • 在while循环外保存curNext指针,初始值可能为nullptr
classSolution{public: ListNode*reverseList(ListNode* head){// 空链表的情况if(head ==nullptr)returnnullptr; ListNode* curNode = head; ListNode* curPrev =nullptr; ListNode* curNext = head->next;// 链表只有一个结点的情况if(curNext ==NULL)return head;while(curNode){ curNode->next = curPrev; curPrev = curNode; curNode = curNext;// curNext可能为空,需要判断 非空时再向后移动if(curNext) curNext = curNext->next;}return curPrev;}};
  • 在while循环外保存curNext指针:不涉及对curNext的解引用,即使为空也不影响
// 思路一、反转指针classSolution{public: ListNode*reverseList(ListNode* head){if(head ==nullptr)returnnullptr; ListNode* curNode = head; ListNode* curPrev =nullptr;while(curNode){// 在 while 循环内保存 curNext,保证了 curNode 一定不为空,避免了对空指针解引用// 可以保证一定可以取到 curNext ,为空或非空 ListNode* curNext = curNode->next; curNode->next = curPrev; curPrev = curNode; curNode = curNext;}return curPrev;}};

思路二:取原链表中的节点,头插到新链表

  • 取链表中的节点,头插到新链表
// 思路二、取链表中的节点,头插到新链表classSolution{public: ListNode*reverseList(ListNode* head){ ListNode* newHead =nullptr,*curNode = head;while(curNode){// 提前保存 curNode 的下一个结点 ListNode* curNext = curNode->next;// 头插 curNode->next = newHead; newHead = curNode;// curNode 向后移动 curNode = curNext;}return newHead;}};

试错代码

  • 初次尝试时的错误代码:
  • 错误原因:
    • 第一次头插后,链表直接断开了,curNode无法移动到下一个结点
// 思路二 取结点,头插到新链表中classSolution{public: ListNode*reverseList(ListNode* head){ ListNode* newHead =nullptr; ListNode* curNode = head;while(curNode){// 错误原因,第一次头插后,链表直接断开了,curNode无法移动到下一个结点if(newHead ==nullptr){ newHead = curNode; newHead->next =nullptr;}else{ curNode->next = newHead; newHead = curNode;} curNode = curNode->next;}return newHead;}};
  • 纠错后的代码
// 思路二 取结点,头插到新链表中classSolution{public: ListNode*reverseList(ListNode* head){ ListNode* newHead =nullptr; ListNode* curNode = head;while(curNode){ ListNode* curNext = curNode->next;// 提前保存 下一个结点if(newHead ==nullptr){ newHead = curNode; newHead->next =nullptr;}else{ curNode->next = newHead; newHead = curNode;} curNode = curNext;}return newHead;}};

算法代码优化

思路一优化:

  • 优化点:无需单独判断链表为空时的情况
    • 链表为空时不进入while循环,直接返回 curPrev,而 curPrev 初始值为 nullptr
// 思路一、反转指针classSolution{public: ListNode*reverseList(ListNode* head){ ListNode* curNode = head,*curPrev =nullptr;while(curNode){// 在 while 循环内保存 curNext,保证了 curNode 一定不为空,避免了对空指针解引用// 可以保证一定可以取到 curNext ,为空或非空 ListNode* curNext = curNode->next; curNode->next = curPrev; curPrev = curNode; curNode = curNext;}return curPrev;}};
  • 思路二已经足够简洁正确,无需优化

以上就是本文的所有内容了,如果觉得文章对你有帮助,欢迎 点赞⭐收藏 支持!如有疑问或建议,请在评论区留言交流,我们一起进步

分享到此结束啦
一键三连,好运连连!
你的每一次互动,都是对作者最大的鼓励!征程尚未结束,让我们在广阔的世界里继续前行! 🚀

Read more

Flutter 三方库 pigeon_generator 的鸿蒙化适配指南 - 在鸿蒙系统上构建极致、自动化的桥接代码生成引擎

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 pigeon_generator 的鸿蒙化适配指南 - 在鸿蒙系统上构建极致、自动化的桥接代码生成引擎 在鸿蒙(OpenHarmony)系统的原生能力调用(如访问鸿蒙特定的硬件传感器、分布式软总线等)中,如何确保 Dart 端与鸿蒙原生(ArkTS/C++)端之间的通讯既高效又具备强类型约束?pigeon_generator 为开发者提供了一套工业级的、基于代码生成的桥接(Platform Channel)自动化方案。本文将深入实战其在鸿蒙原生适配中的应用。 前言 什么是 Pigeon Generator?它是 Flutter 官方推荐的类型安全通讯工具。传统的 MethodChannel 依赖于动态字符串映射,在鸿蒙开发这种多模块、复杂业务场景下,极易产生拼写错误或类型不匹配。pigeon_generator 通过定义一套统一的协议文件,自动生成

By Ne0inhk
鸿蒙电商购物车全栈项目——用户增长、性能优化、Next原生合规

鸿蒙电商购物车全栈项目——用户增长、性能优化、Next原生合规

《鸿蒙APP开发从入门到精通》第15篇:鸿蒙电商购物车全栈项目——用户增长、性能优化、Next原生合规 📈⚡✅ 内容承接与核心价值 这是《鸿蒙APP开发从入门到精通》的第15篇——用户增长、性能优化、Next原生合规篇,100%承接第14篇的「订单管理、支付管理、AI原生」项目架构,完成鸿蒙电商购物车全栈项目的性能优化与合规性调整。 学习目标: * 掌握用户增长的设计与实现; * 实现用户行为分析、用户留存优化、用户转化提升; * 理解性能优化的设计与实现; * 实现启动优化、渲染优化、网络优化; * 掌握Next原生合规的设计与实现; * 实现代码规范、权限合规、数据合规; * 优化用户增长、性能优化、Next原生合规的用户体验(响应速度、数据安全、用户反馈)。 学习重点: * 鸿蒙APP用户增长的开发流程; * 用户增长的分类与使用场景; * 性能优化的设计与实现; * Next原生合规的设计与实现。 一、 用户增长基础 🎯 1.1 用户增长定义 用户增长是指通过各种手段提升用户的数量、

By Ne0inhk
Flutter 组件 fluid_layout 的适配 鸿蒙Harmony 实战 - 驾驭全场景动态自适应栅格、实现鸿蒙端弹性布局分发与多端显示适配方案

Flutter 组件 fluid_layout 的适配 鸿蒙Harmony 实战 - 驾驭全场景动态自适应栅格、实现鸿蒙端弹性布局分发与多端显示适配方案

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 组件 fluid_layout 的适配 鸿蒙Harmony 实战 - 驾驭全场景动态自适应栅格、实现鸿蒙端弹性布局分发与多端显示适配方案 前言 在鸿蒙(OpenHarmony)生态的“一次开发、多端部署”战略中,面对需要在华为手机、MatePad、智慧屏、甚至车载大屏等不同分辨率、不同宽纵比的设备间无缝流转的 UI 设计。如果仅仅依靠写死的 double 宽度或者是简单的 MediaQuery.of(context).size。那么不仅会导致在折叠屏(Foldable)展开瞬间产生严重的界面坍塌,更会因为缺乏一套工业级的栅格(Grid)规范。引发在不同 DPI 下文字重叠、按钮溢出以及留白失控等严重的适配事故方案。 我们需要一种“流动感知、栅格克制”的布局艺术。

By Ne0inhk

centos stream 10 系统下 Docker 安装与配置全流程指南

centos stream 10 系统下 Docker 安装与配置全流程指南 文章目录 * centos stream 10 系统下 Docker 安装与配置全流程指南 * 一、前置条件 * 1.1 系统要求 * 二、安装流程 * 步骤 1:卸载旧版本 Docker(如有) * 步骤 2:安装依赖工具 * 步骤 3:配置 Docker 仓库(推荐国内镜像源) * 选项 1:阿里云镜像源(推荐) * 选项 2:清华大学镜像源 * 步骤 4:安装 Docker 引擎 * 步骤 5:启动 Docker

By Ne0inhk