leetcodeHot100|148.排序链表

排序链表

给你链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表 。

示例 1:

输入:head = [4,2,1,3]

输出:[1,2,3,4]

示例 2:

输入:head = [-1,5,3,4,0]

输出:[-1,0,3,4,5]

示例 3:

输入:head = []

输出:[]

提示:

链表中节点的数目在范围 [0, 5 * 104] 内

-105 <= Node.val <= 105

思路分析

首先需要理解题目,题目是给出了一个单链表,需要你去把它排序之后给出来。 核心思路是分而治之,把复杂问题变为小问题解决,再合并解决。

关键是如何去拆链表呢?用快慢指针去找中点,慢指针之走一步,快指针走两步,快指针快到末尾的时候,慢指针就在中点。比如在这个图书中的中点就是5.

如何去合并呢?使用虚拟头节点去凭借两个有序的链表就可以,每次选着一个更小的节点连接上去。

最终演示

cpp 复制代码
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* sortList(ListNode* head) {
        // 归并排序入口:排序范围是 [head, nullptr)(即从head到链表末尾)
        return sortList(head, nullptr);
    }

    // 递归函数:排序范围是 [head, tail)(左闭右开,tail是终止边界,不包含)
    ListNode* sortList(ListNode* head, ListNode* tail) {
        // 错误1:赋值号= 改成 判断相等==(核心语法错误)
        // 边界条件1:如果当前区间只有0个节点,直接返回null
        if (head == nullptr) {
            return head;
        }
        // 错误2:赋值号= 改成 判断相等==(核心语法错误)
        // 边界条件2:如果当前区间只有1个节点(head->next == tail),断开后续链接并返回
        if (head->next == tail) {
            head->next = nullptr; // 断开与tail的链接,避免后续merge时出现环
            return head;
        }
        // 快慢指针找中间节点(mid):慢指针走1步,快指针走2步
        ListNode* slow = head, *fast = head;
        // 循环条件:快指针未到边界tail
        while (fast != tail) {
            slow = slow->next;       // 慢指针走1步
            fast = fast->next;       // 快指针走1步
            if (fast != tail) {      // 快指针未到边界时,再走1步(总共2步)
                fast = fast->next;
            }
        }
        ListNode* mid = slow; // 找到中间节点,将链表分为 [head, mid) 和 [mid, tail)
        // 递归排序左半部分和右半部分,然后合并
        return merge(sortList(head, mid), sortList(mid, tail));
    } 

    // 合并两个已排序的链表,返回合并后的头节点
    ListNode* merge(ListNode* head1, ListNode* head2) { 
        ListNode* dummyHead = new ListNode(0); // 虚拟头节点(简化链表拼接逻辑)
        ListNode* temp = dummyHead;    // 临时指针,用于构建新链表
        ListNode* temp1 = head1;       // 遍历第一个链表的指针
        ListNode* temp2 = head2;       // 遍历第二个链表的指针
        
        // 循环:同时遍历两个链表,每次取较小值的节点拼接到新链表
        while (temp1 != nullptr && temp2 != nullptr) {  
            if (temp1->val <= temp2->val) { 
                temp->next = temp1;    // 拼接temp1节点
                temp1 = temp1->next;   // temp1后移
            } else { 
                temp->next = temp2;    // 拼接temp2节点
                temp2 = temp2->next;   // temp2后移
            }
            temp = temp->next;         // 临时指针后移,准备拼接下一个节点
        }
        // 处理剩余节点:如果第一个链表还有剩余,直接拼接
        if (temp1 != nullptr) { 
            temp->next = temp1;
        } 
        // 处理剩余节点:如果第二个链表还有剩余,直接拼接(注意这里用else即可,无需else if)
        else {
            temp->next = temp2;
        }
        // 返回合并后链表的真实头节点(跳过虚拟头节点)
        return dummyHead->next;
    }
};

遇到的问题

使用虚拟头结点dummyHead可以不用处理空链表的情况,一定要学会使用,还是就是每次接完一个节点,指针必须往后走,不然容易陷入死循环。最后一个节点也需要注意贴到最后.

总结

链表归并排序的核心是分治,拆分成小链表然后排序再合并其中合并两个有序的链表是基础.

需要项目中图解的可以私信我哦

相关推荐
漂流瓶jz几秒前
UVA-11846 找座位 题解答案代码 算法竞赛入门经典第二版
数据结构·算法·排序算法·深度优先·aoapc·算法竞赛入门经典·uva
py有趣1 小时前
力扣热门100题之合并两个有序链表
算法·leetcode·链表
东北甜妹2 小时前
MYSQL 总结
数据结构
CheerWWW2 小时前
C++学习笔记——this关键字、对象生命周期(栈作用域)、智能指针、复制与拷贝构造函数
c++·笔记·学习
lucky九年2 小时前
GO语言模拟C++封装,继承,多态
开发语言·c++·golang
北顾笙9802 小时前
day12-数据结构力扣
数据结构·算法·leetcode
漫随流水2 小时前
c++编程:D进制的A+B(1022-PAT乙级)
数据结构·c++·算法
tankeven3 小时前
HJ159 没挡住洪水
c++·算法
charlie1145141913 小时前
嵌入式C++教程实战之Linux下的单片机编程:从零搭建 STM32 开发工具链(5):调试进阶篇 —— 从 printf 到完整 GDB 调试环境
linux·c++·单片机·学习·嵌入式·c
paeamecium3 小时前
【PAT】 - Course List for Student (25)
数据结构·c++·算法·pat考试