LeetCode148.排序链表

看完题目的想法是,直接把所有节点的值都遍历出来放进优先队列里面,然后从头节点遍历一次,每次把优先队列poll()的值赋给节点的val即可,说实话,想完还觉得估计有问题怎么可能这么简单,但是不管了,5分钟就把这个算法写出来了,一提交,居然通过了!以下是我的代码:

java 复制代码
class Solution {
    public ListNode sortList(ListNode head) {

        PriorityQueue<Integer> pri = new PriorityQueue<>(new Comparator<Integer>(){
            public int compare(Integer e1, Integer e2){
                  return e1 - e2;
            }
        });

      ListNode h = head;
      while(h != null){
          pri.add(h.val);
          h = h.next;
      }
      ListNode h2 = head;
      while(h2 != null){
          h2.val = pri.poll();
          h2 = h2.next;
      }
      
     return head;

    }

其实这个优先队列也不用new一个比较器实例,因为默认是从小到大的。然后看看官方题解吧,不要用我这种二流子写法了。

题解用的是归并排序,先把链表分成两半,每半分别排序,然后再把排完序的两半合并起来;对于两半中的每一半也是这样的,把这半再分成两半,两半分别排好序合起来,只有当"一半"只有两个节点是不用再分,直接比较这两个节点然后排序,然后再与另一半合起来,然后再与更大的另一半合起来...一直合到这个完整的最大的链表。

分割可以采用快慢指针的方法,快慢指针同时从头节点出发,快指针每次走两步,慢指针每次走一步,当快指针到达链尾,慢指针就在中间节点。然后利用递归的方法不断的分割链表,直到只剩两个节点,开始合并。

合并先创建一个哑节点,然后分别比较左右两个链表的头节点,最小的先移到哑节点后面,然后这个链表的指针移到下一个节点,下次比较就是这个链表的第2个节点和另一个链表的第一个节点,(因为两个链表都是已经排好序的,所以每次只要比较两个链表未放进去的最小节点即可),如果一个链表已经遍历完了,只要把另一个链表剩下的部分直接挂在后面即可。

以下是题解代码:

java 复制代码
class Solution {
    public ListNode sortList(ListNode head) {
        return sortList(head, null);
    }

    public ListNode sortList(ListNode head, ListNode tail) {
        if (head == null) {
            return head;
        }
        if (head.next == tail) {
            head.next = null;
            return head;
        }
        ListNode slow = head, fast = head;
        while (fast != tail) {
            slow = slow.next;
            fast = fast.next;
            if (fast != tail) {
                fast = fast.next;
            }
        }
        ListNode mid = slow;
        ListNode list1 = sortList(head, mid);
        ListNode list2 = sortList(mid, tail);
        ListNode sorted = merge(list1, list2);
        return sorted;
    }

    public ListNode merge(ListNode head1, ListNode head2) {
        ListNode dummyHead = new ListNode(0);
        ListNode temp = dummyHead, temp1 = head1, temp2 = head2;
        while (temp1 != null && temp2 != null) {
            if (temp1.val <= temp2.val) {
                temp.next = temp1;
                temp1 = temp1.next;
            } else {
                temp.next = temp2;
                temp2 = temp2.next;
            }
            temp = temp.next;
        }
        if (temp1 != null) {
            temp.next = temp1;
        } else if (temp2 != null) {
            temp.next = temp2;
        }
        return dummyHead.next;
    }
}
相关推荐
少许极端23 分钟前
算法奇妙屋(十五)-BFS解决边权为1的最短路径问题
数据结构·算法·bfs·宽度优先·队列·图解算法·边权为1的最短路径问题
ada7_1 小时前
LeetCode(python)——94.二叉
python·算法·leetcode·链表·职场和发展
不穿格子的程序员1 小时前
从零开始写算法——普通数组类题:数组操作中的“翻转技巧”与“前后缀分解”
数据结构·算法
逝雪Yuki1 小时前
简单多源BFS问题
算法·leetcode·bfs·广度优先遍历
curry____3031 小时前
study in PTA(高精度算法与预处理)(2025.12.3)
数据结构·c++·算法·高精度算法
代码游侠1 小时前
学习笔记——栈
开发语言·数据结构·笔记·学习·算法
Ayanami_Reii1 小时前
进阶数据结构应用-维护序列
数据结构·算法·线段树
CoderYanger2 小时前
C.滑动窗口-越长越合法/求最短/最小——2904. 最短且字典序最小的美丽子字符串
java·开发语言·数据结构·算法·leetcode·1024程序员节
别动哪条鱼2 小时前
FFmpeg API 数据结构及其详细说明:
数据结构·ffmpeg·音视频·aac
CoderYanger2 小时前
A.每日一题——2141.同时运行N台电脑的最长时间
java·算法·leetcode·职场和发展·1024程序员节