1957

无聊蛋疼的1957写的低端博客
随笔 - 214, 文章 - 1, 评论 - 38, 阅读 - 21万
  博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

[leetcode]Sort List

Posted on   1957  阅读(224)  评论(0编辑  收藏  举报

链表用归并排序,之前也做过,用递归写很简单,代码不长也很好理解。地址

但是今天无聊想用非递归写下。

基本思想就是每次就是1,2,4,8,16这样分块。。然后比较,合并。

想法很简单,结果代码写出来也debug了一小会儿。

主要遇到的问题就是,在merge两个list的时候(其实就是一个的两部分),怎么处理末尾和开始。

没处理好,可能就是个环。

还是在代码里面注视说好点。

 

复制代码
class Solution {
public:
    ListNode *get_next(ListNode* head, int n) {
        while(n-->0 && head) {
            head = head->next;
        }
        return head;
    }
    
    ListNode* merge2(ListNode* l1, ListNode* l2, int n, int pos, int len) {
        ListNode* dummy = new ListNode(-1);
        ListNode* curr = dummy;
        ListNode* le = l2;
        int n1 = min(n, len - pos); // 计算l1有多少个节点
        int n2 = min(n, len - (pos+n)); // 计算l2有多少个节点
        while(n1 > 0 && n2 > 0) { //开始是用l1,l2这样表示的,明显的问题,只是其中一段,next肯定不是nullptr
            if (l1->val < l2->val) {
                curr->next = l1;
                l1 = l1->next;
                n1--;
            } else {
                curr->next = l2;
                n2--;
                l2 = l2->next;
            }
            curr = curr->next;
        }
        if (n1 > 0&& l1) curr->next = l1;
        if (n2 > 0&& l2) curr->next = l2;
        ListNode* ans = dummy->next;
        delete dummy;
        return ans;
    }
    
    ListNode *sortList(ListNode *head) {
        if (head == nullptr || head->next == nullptr) return head;
        int len = 0;
        for (ListNode* tmp = head; tmp; tmp = tmp->next) {
            len++;
        }
        int step = 1; // step = 1,2,4,6,8

        while (step < len ) {
            ListNode* curr = head;
            ListNode* prev = nullptr;
            for (int i = 0; i < len; i += step*2) { //i其实就是curr在list里面的位置
                ListNode* next = get_next(curr, step*2); //记录next...不然前面重排之后不好找
                ListNode* tmp = merge2(curr, get_next(curr, step), step, i, len); // l1 是当前节点的step个,l2是step+1到step*2个
                if (i == 0) head = tmp; //每次迭代head可能都会变
                curr = tmp;
                if (prev) prev->next = curr; //把前面的指向现在的head
                prev = get_next(curr, step * 2 - 1);  //这个是现在l1,l2合并后的最后一个节点,就是后面的prev
                curr = next; //curr指向下一个
            }
            step *= 2;
        }
        ListNode* curr = get_next(head, len - 1);//最后的节点的next其实可能还指向了某些地方的 2->1这种,我们排序之后是1->2但是2还是指向1的,就有环
        curr->next = nullptr;
        return head;
    }
    
};
复制代码

 

编辑推荐:
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
阅读排行:
· 全程不用写代码,我用AI程序员写了一个飞机大战
· DeepSeek 开源周回顾「GitHub 热点速览」
· 记一次.NET内存居高不下排查解决与启示
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
历史上的今天:
2014-01-16 [leetcode]Text Justification
2014-01-16 [leetcode]Sqrt(x)
点击右上角即可分享
微信分享提示