news 2026/6/10 13:56:01

单链表逆转

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
单链表逆转

一、题目要求

1.1 需求与接口

实现函数List Reverse( List L ),输入单链表头指针L,返回逆转后的链表头指针。

1.2 数据结构定义

typedef int ElementType; typedef struct Node *PtrToNode; struct Node { // 节点结构:数据域+指针域 ElementType Data; PtrToNode Next; }; typedef PtrToNode List; // List等价于指向节点的指针,代表链表头指针

1.3 输入输出样例

  • 输入:5 1 3 4 5 2(5 为节点个数,后续数字为各节点数据)
  • 输出:1(原链表头节点);2 5 4 3 1(逆转后的链表)

二、题目分析

2.1 链表结构

题目中链表为不带头节点类型,即头指针L直接指向第一个数据节点,节点间通过Next指针串联,末尾节点的NextNULL

以输入样例为例,链表结构可视化:L → 节点1(Data=1, Next→3) → 节点3(Data=3, Next→4) → 节点4→节点5→节点2→NULL

2.2 解题思路

单链表逆转的本质是重新调整节点指针的指向,将每个节点的Next指针从 “指向下一个节点” 改为 “指向前一个节点”。常用两种实现思路:

  1. 迭代法(头插法):逐个摘下原链表的节点,将其插入到新链表的头部,最终新链表即为逆转结果。
  2. 递归法:采用分治思想,先递归逆转当前节点之后的子链表,再调整当前节点与子链表的指针关系。

三、答案及解析

3.1 迭代法(推荐)

List Reverse( List L ){ List p, q; // p为遍历指针,q暂存p的后继节点(防止断链) List newHead = NULL; // 逆转链表的头指针(核心变量) p = L; // 从原链表第一个节点开始遍历 while(p != NULL){ q = p->Next; // 1. 保存p的后继,避免后续操作丢失原链表后续节点 p->Next = newHead; // 2. 将p节点挂到逆转链表的头部 newHead = p; // 3. 更新逆转链表的头指针为p p = q; // 4. p移动到下一个待处理节点 } return newHead; // 返回逆转后的链表头指针 }
解析
  1. 变量

    • newHead:始终指向已完成逆转的链表部分的头节点,初始值为NULL(表示逆转链表为空),是整个算法的核心锚点。
    • q:用于暂存p的后继节点,因为修改p->Next后会丢失原链表的后续节点,必须提前保存。
    • p:遍历原链表的指针,逐个处理每个节点。
  2. newHead 的完整变化流程

  3. 以输入样例1→3→4→5→2为例,将newHead的变化与逆转链表状态对应,清晰展示每一步操作:

    处理节点newHead 初始值p->Next = newHead(连接)newHead = p(更新)逆转链表状态newHead 的含义
    无(初始)NULL--逆转链表为空
    节点 1NULL节点 1→Next = NULLnewHead→节点 11→NULL逆转链表的头是节点 1
    节点 3节点 1节点 3→Next = 节点 1newHead→节点 33→1→NULL逆转链表的头是节点 3
    节点 4节点 3节点 4→Next = 节点 3newHead→节点 44→3→1→NULL逆转链表的头是节点 4
    节点 5节点 4节点 5→Next = 节点 4newHead→节点 55→4→3→1→NULL逆转链表的头是节点 5
    节点 2节点 5节点 2→Next = 节点 5newHead→节点 22→5→4→3→1→NULL逆转链表的头是节点 2
  4. 终止与返回p=NULL时,说明原链表遍历完成,此时newHead指向原链表的最后一个节点(样例中为节点 2),也就是逆转后链表的头节点,返回newHead即可完成逆转。

3.2 递归法

List Reverse( List L ){ // 终止条件:空链表或只有一个节点的链表,无需逆转直接返回 if (L == NULL || L->Next == NULL) return L; // 1. 递归逆转子链表(处理L->Next开始的部分) List newList = Reverse(L->Next); // 2. 调整当前节点与子链表的指针关系 L->Next->Next = L; // 子链表的末尾节点指向当前节点L L->Next = NULL; // 切断L与原后继的联系,防止链表成环 // 3. 返回逆转后的总链表头(始终是原链表的末尾节点) return newList; }
解析
  1. 终止条件当链表为空(L=NULL)或只有一个节点(L->Next=NULL)时,无需逆转,直接返回L,这是递归的出口,避免无限递归。

  2. 递归核心以样例链表1→3→4→5→2为例:

    • 调用Reverse(L)时,会先递归调用Reverse(3)Reverse(3)又调用Reverse(4),直到Reverse(2)触发终止条件,返回节点 2(newList=2)。
    • 从最深层递归返回,依次调整每个节点的指针。
  3. 指针调整

    • 当回溯到Reverse(5)时:L=5L->Next=2,执行L->Next->Next = L2->Next=5,再执行L->Next=NULL5->Next=NULL,此时子链表为2→5→NULL,返回newList=2
    • 继续回溯,依次调整节点 4、3、1 的指针,最终得到逆转链表2→5→4→3→1→NULL
  4. 返回值递归过程中,newList始终指向原链表的末尾节点(逆转后的头节点),最终返回该节点即可。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 13:29:38

Dify 1.7.0音频功能瓶颈突破(音频时长限制终极应对策略)

第一章:Dify 1.7.0音频功能瓶颈突破(音频时长限制终极应对策略)Dify 1.7.0 版本在语音处理能力上实现了显著增强,但仍存在单次音频上传时长上限为60秒的硬性限制。这一约束对需要处理长语音的应用场景构成挑战。通过合理的技术拆分…

作者头像 李华
网站建设 2026/6/10 3:41:40

如何通过vivado对一个FPGA工程进行性能评估

目录 1.最高运行频率(Fmax​) 2.资源利用率 3.功耗(Power) 4.传输延迟(Latency) 5.吞吐率(Throughput) 在开展FPGA设计的性能评估工作时,需围绕多个核心维度展开量化分析,常用的关键评估指标主要包含以下五类: 最高运行频率(Fmax​)&…

作者头像 李华
网站建设 2026/6/9 13:04:14

自定义类或结构体-–-behaviac

原文 在.h文件中,任意编写一个自定义的类或结构体,并用宏DECLARE_BEHAVIAC_STRUCT声明该类或者结构体为非虚类,如下代码所示: struct TypeTest2_t {int name;float weight;bool bLive;DECLARE_BEHAVIAC_STRUCT(TypeTest2_t);…

作者头像 李华
网站建设 2026/6/9 23:52:31

清朝条约全集 PDF 电子版(三册合集):从尼布楚到辛丑条约的完整史料集

若想系统梳理清代对外关系的史料脉络,这份《清代外交文献三册合辑》可作为重要参考 —— 它完整收录了自 1689 年《尼布楚条约》至 1901 年《辛丑条约》期间的核心外交文书,覆盖清代近 220 年的对外交往历程。 这套合辑以时间为线索分册编排&#xff1a…

作者头像 李华
网站建设 2026/6/10 12:53:16

6、数据保护:备份、恢复与业务连续性

数据保护:备份、恢复与业务连续性 在当今数字化时代,数据保护对于各类组织而言至关重要。它不仅关乎数据的安全性,还涉及业务的连续性和灾难恢复能力。本文将深入探讨数据保护的各个方面,包括面临的挑战、可用的机会以及具体的保护策略和技术。 1. 数据保护基础 在 IT 领…

作者头像 李华
网站建设 2026/6/10 6:02:34

检索重排序的 Dify 结果过滤:5步实现精准结果过滤与排序优化

第一章:检索重排序的 Dify 结果过滤在构建基于大语言模型的应用时,检索增强生成(RAG)系统常面临检索结果相关性不足的问题。Dify 作为低代码 AI 应用开发平台,提供了灵活的结果过滤与重排序机制,帮助开发者…

作者头像 李华