Leetcode——链表:143.重排链表

题目

思路

首先考虑特殊情况,链表为空,或者链表只有一个元素,此时直接返回

找到中间位置,将后半部分的链表翻转,得到新链表,将后半部分链表的节点交替插入原链表

寻找链表中间节点

使用快慢指针法

设置一个快指针,一个慢指针,初始均指向链表头节点

慢指针一次向后移动一个节点,快指针一次向后移动两个节点

当快指针移动到结尾时,慢指针指向前半条链表的最后一个节点

设置一个中间节点,中间节点为慢指针的下一个

断开前后两部分链表,慢指针的下一个指向空

原地翻转链表

设置三个变量prev,curr,next

设置循环处理全部链表中的节点,将当前元素的后继改为原链表的前驱(此时当前元素与原链表的后继断开,造成后继链表中全部节点丢失)

更新三个变量,处理下一个节点,将当前元素变为prev,将next改为当前元素,将next向后移动一个节点

当移动到最后一个元素时,把当前节点直接插入到原链表的头,更新并返回新链表的头

合并两个链表

设置新链表的尾指针,初始值设为指向第一个链表的头节点

将第一条链表的头指针设为链表中的第二个节点

先将第二个链表的头节点插入第一个链表中,尾向后移动,第二个链表的头节点向后移动

将第一个链表的第二个节点连接到尾后,尾向后移动,第一个节点指针向后移动

代码

寻找链表中间节点

传入值:链表头节点

传出值:后半部分链表的头节点

复制代码
ListNode* midList(ListNode* head)
    {
        ListNode* fast=head;
        ListNode* slow=head;

        while(fast->next!=nullptr&&fast->next->next!=nullptr)
        {
            fast=fast->next->next;
            slow=slow->next;
        }
        ListNode* MidNode=slow->next;
        slow->next=nullptr;
        return MidNode;
    }

原地翻转链表

传入值:链表头节点

传出值:新链表头节点

复制代码
ListNode* reverseList(ListNode* head)
    {
        ListNode* prev=nullptr;
        ListNode* curr=head;
        ListNode* next=curr->next;

        while(next!=nullptr)
        {
            curr->next=prev;

            prev=curr;
            curr=next;
            next=next->next;
        }

        curr->next=prev;
        prev=curr;

        return prev;
    }

交替合并两个链表

复制代码
void mergeList(ListNode* list1_head,ListNode* list2_head)
    {
        ListNode* newListTail=list1_head;
        list1_head=list1_head->next;
        while(list1_head!=nullptr&&list2_head!=nullptr)
        {
            newListTail->next=list2_head;
            newListTail=newListTail->next;
            list2_head=list2_head->next;

            newListTail->next=list1_head;
            newListTail=newListTail->next;
            list1_head=list1_head->next;
        }
        newListTail->next=list2_head;
    }

主函数

复制代码
void reorderList(ListNode* head) {
        if(head==nullptr||head->next==nullptr)return;

        ListNode* secondList=midList(head);
        secondList=reverseList(secondList);
        mergeList(head,secondList);
    }
相关推荐
莹莹学编程—成长记1 小时前
string的模拟实现
服务器·c++·算法
Minyy111 小时前
SpringBoot程序的创建以及特点,配置文件,LogBack记录日志,配置过滤器、拦截器、全局异常
xml·java·spring boot·后端·spring·mybatis·logback
画个大饼3 小时前
Go语言实战:快速搭建完整的用户认证系统
开发语言·后端·golang
喵先生!4 小时前
C++中的vector和list的区别与适用场景
开发语言·c++
xMathematics5 小时前
计算机图形学实践:结合Qt和OpenGL实现绘制彩色三角形
开发语言·c++·qt·计算机图形学·cmake·opengl
ShiinaMashirol5 小时前
代码随想录打卡|Day27(合并区间、单调递增的数字、监控二叉树)
java·算法
yuanManGan7 小时前
C++入门小馆: 深入了解STLlist
开发语言·c++
梁下轻语的秋缘7 小时前
每日c/c++题 备战蓝桥杯(P1049 [NOIP 2001 普及组] 装箱问题)
c语言·c++·学习·蓝桥杯
逐光沧海7 小时前
STL常用算法——C++
开发语言·c++
wuqingshun3141597 小时前
蓝桥杯 5. 交换瓶子
数据结构·c++·算法·职场和发展·蓝桥杯