当前位置:首页 > 力扣 > 力扣第92题:三步定位 精准反转链表指定区间

力扣第92题:三步定位 精准反转链表指定区间

10个月前 (05-19)

力扣第92题:三步定位 精准反转链表指定区间 力扣 C++ 栈 链表 数据结构 算法 第1张题目解读

给定一个单链表和两个整数left与right,要求将链表中从第left个节点到第right个节点的部分进行反转,而保持其他部分不变。例如,对于链表1→2→3→4→5,left=2,right=4,反转后应为1→4→3→2→5。这个问题考察了对链表操作的熟练程度,特别是如何在不破坏链表整体结构的情况下,精确地对指定区间进行反转。


思路与过程

1.用栈结构辅助完成链表部分反转的操作。首先处理特殊情况,当left等于right时直接返回原链表。然后通过遍历链表定位四个关键节点:leftnode(反转区间起始节点)、leftlast(反转区间前一个节点)、rightnode(反转区间结束节点)和rightnext(反转区间后一个节点)。

2.找到这些关键节点后,将需要反转的区间节点依次压入中。然后根据left是否为1(即是否从链表头开始反转)分别处理:如果从头部开始反转,则更新链表头;否则将leftlast的next指向栈顶节点。最后依次弹出栈中节点完成反转,并将反转后的最后一个节点与rightnext连接起来。


代码与注释

class Solution {
public:
    ListNode* reverseBetween(ListNode* head, int left, int right) {
        if(left==right) // 特殊情况处理:不需要反转
        {
            return head;
        }
        // 定义四个关键节点指针
        ListNode* leftnode;  // 反转区间起始节点
        ListNode* leftlast;  // 反转区间前一个节点
        ListNode* rightnode; // 反转区间结束节点
        ListNode* rightnext; // 反转区间后一个节点
        ListNode* tmp=head;  // 临时指针用于遍历
        
        // 遍历链表定位关键节点
        for(int i=1;i<=right+1;i++)
        {
            if(i==left)
            {
                leftnode=tmp; // 记录反转起始节点
            }
            if(i==left-1)
            {
                leftlast=tmp; // 记录反转前一个节点
            }
            if(i==right)
            {
                rightnode=tmp; // 记录反转结束节点
            }
            if(i==right+1)
            {
                if(tmp!=rightnode)
                {rightnext=tmp;} // 记录反转后一个节点
                else
                {rightnext=nullptr;} // 处理反转到链表末尾的情况
            }
            if(tmp->next!=nullptr)
                tmp=tmp->next; // 移动指针
        }
        
        // 使用栈存储需要反转的节点
        stack<ListNode*> stk;
        while(leftnode!=rightnode->next)
        {
            stk.push(leftnode); // 压入反转区间节点
            leftnode=leftnode->next;
        }

        // 处理反转后的连接
        if(left==1) // 从链表头开始反转的情况
        {
            tmp=stk.top();
            stk.pop();
            head=tmp; // 更新链表头
        }
        else // 中间部分反转的情况
        {
            tmp=leftlast;
            tmp->next=stk.top(); // 连接反转区间前节点与反转后的第一个节点
            tmp=tmp->next;
            stk.pop();
        }

        // 完成剩余节点的反转连接
        while(!stk.empty())
        {
            tmp->next=stk.top(); // 连接反转后的节点
            tmp=tmp->next;
            stk.pop();
        }
        
        // 处理反转区间后的连接
        if(rightnext!=nullptr)
        {
            tmp->next=rightnext; // 连接反转后的最后一个节点与后续节点
        }
        else{
            tmp->next=nullptr; // 处理反转到链表末尾的情况
        }

        return head;
    }
};


原创内容 转载请注明出处

分享给朋友:

相关文章

力扣53题:贪心策略与动态规划的完美联姻 三行代码映射算法精髓

力扣53题:贪心策略与动态规划的完美联姻 三行代码映射算法精髓

题目理解在数字的海洋中寻找最具价值的珍珠链:当我们面对一个可能包含正负数的数组时,寻找连续子数组的和最大值就像在波动的股票曲线中捕捉最佳投资时段。问题的核心在于如何处理可能降低总和的负值元素——是忍痛...

力扣第71题:用栈轻松解决Unix路径简化问题

力扣第71题:用栈轻松解决Unix路径简化问题

题目解读:在Unix风格的文件系统中,我们经常需要处理各种复杂的路径表示。给定一个绝对路径字符串,我们需要将其转换为最简化的规范路径。规范路径要求:路径始终以斜杠'/'开头;两个目录名...

力扣654:递归分治的艺术 如何用最大元素构建二叉树

力扣654:递归分治的艺术 如何用最大元素构建二叉树

题目重解我们面对一个看似简单却充满递归魅力的题目:给定一个不含重复元素的整数数组,需要构建一棵特殊的二叉树。这个树的每个父节点都必须是当前子数组中的最大元素,而它的左右子树则分别由该最大值左侧和右侧的...

力扣933题:队列的妙用:如何高效统计最近请求

力扣933题:队列的妙用:如何高效统计最近请求

题目重解:我们需要设计一个能统计最近3000毫秒内请求次数的系统。每当新的请求到来时,它会带有时间戳t,我们需要返回过去3000毫秒内(包括当前)发生的请求总数。这就像是在时间轴上维护一个滑动窗口,只...

牛客14496题解:括号最大深度问题(栈思想与代码优化)

牛客14496题解:括号最大深度问题(栈思想与代码优化)

一、题目解读牛客14496题要求计算给定括号字符串中的最大深度。例如,对于字符串 "(()())",最大深度为2。题目考察对括号嵌套结构的理解,以及如何通过编程找到最深嵌套层次。二...

洛谷1220题解:动态规划与区间DP优化解法(附代码注释)

洛谷1220题解:动态规划与区间DP优化解法(附代码注释)

一、题目解读洛谷1220题要求计算在n个位置放置灯的情况下,通过关闭连续区间灯并移动至区间端点,使得总耗电量最小。需考虑灯的功率与位置差异,设计高效的算法求解最优策略。二、解题思路1. 动态规划 +...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。