IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 148.排序链表【leetcode】 -> 正文阅读

[数据结构与算法]148.排序链表【leetcode】

题目

在这里插入图片描述
在这里插入图片描述
字节面试题

复杂度分析

时间复杂度O(nlogn)
空间复杂度为O(1)

解题思路参照官方做法2:自底向上归并排序
这里给出注释版,便于本次和以后做这题的理解
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:
//官方做法2:自底向上归并排序
    ListNode* sortList(ListNode* head) {
        if(head==nullptr) return head;

        // 1. 首先从头向后遍历,统计链表长度
        int length = 0; // 用于统计链表长度
        ListNode* node = head;
        while(node!=nullptr){
            length++;
            node = node->next;
        }

        // 2. 初始化 引入dummynode
        ListNode * dummyHead = new ListNode(0);
        dummyHead->next = head;

        // 3. 每次将链表拆分成若干个长度为subLen的子链表 , 并按照每两个子链表一组进行合并
        for(int subLen=1;subLen<length;subLen=subLen*2){
            ListNode* prev = dummyHead;
            ListNode* curr = dummyHead->next;// curr用于记录拆分链表的位置

            while(curr!=nullptr){   // 如果链表没有被拆完
                // 3.1 拆分subLen长度的链表1
                ListNode* head1 = curr;     // 第一个链表的头 即 curr初始的位置
                for(int i=1;i<subLen&&curr!=nullptr&&curr->next!=nullptr;i++){// 拆分出长度为subLen的链表1
                    curr = curr->next;
                }

                // 3.2 拆分subLen长度的链表2
                ListNode* head2 = curr->next;   // 第二个链表的头  即 链表1尾部的下一个位置
                curr->next = nullptr;           // 断开第一个链表和第二个链表的链接
                curr = head2;                   // 第二个链表头 重新赋值给curr
                for(int i=1;i<subLen&&curr!=nullptr&&curr->next!=nullptr;++i){// 再拆分出长度为subLen的链表2
                    curr = curr->next;
                }

                // 3.3 再次断开 第二个链表最后的next的链接
                ListNode* next = nullptr;
                if(curr!=nullptr){
                    next = curr->next;      // next用于记录 拆分完两个链表的结束位置
                    curr->next = nullptr;   // 断开连接
                }

                // 3.4 合并两个subLen长度的有序链表
                ListNode* merged = mergeTwoLists(head1,head2);
                prev->next = merged;        // prev->next 指向排好序链表的头
                while(prev->next!=nullptr){ // while循环 将prev移动到 subLen*2 的位置后去
                    prev = prev->next;
                }
                curr = next;                // next用于记录 拆分完两个链表的结束位置
            }

        }
        // 返回新排好序的链表
        return dummyHead->next;
    }

    //合并两个链表
    ListNode* mergeTwoLists(ListNode* l1,ListNode* l2){
        ListNode* dummy = new ListNode(0);
        ListNode* curr = dummy;
        while(l1!=nullptr && l2!=nullptr){  // 退出循环的条件是走完了其中一个链表
            // 判断l1 和 l2大小
            if(l1->val < l2->val){
                // l1 小 , curr指向l1
                curr->next = l1;
                l1 = l1->next;  // l1 向后走一位
            }else{
                // l2 小 , curr指向l2
                curr->next = l2;
                l2 = l2->next;  // l2向后走一位
            }
            curr = curr->next;  // curr后移一位
        }

        // 退出while循环之后,比较哪个链表剩下长度更长,直接拼接在排序链表末尾
        if(l1==nullptr) curr->next = l2;
        if(l2==nullptr) curr->next = l1;

        // 最后返回合并后有序的链表
        return dummy->next;
    }
};
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-05-14 10:09:04  更:2022-05-14 10:10:19 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/1 23:37:09-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码