排序链表的三种写法

题目链接:https://leetcode.cn/problems/sort-list/?envType=study-plan-v2&envId=top-100-liked

第一种,插入排序,会超时

java 复制代码
class Solution {
    public ListNode sortList(ListNode head) {
        //插入排序,用较为简单的方式解决
        ListNode dummyNode = new ListNode(0,null);
        ListNode p = head;
        while(p!=null){
            ListNode temp = p;
            p=p.next;
            //将temp插入到链表当中
            ListNode q = dummyNode;
            int flag = 0;
            while(q.next!=null){
                if(q.next.val>=temp.val){
                    ListNode t = q.next;
                    q.next = temp;
                    temp.next = t;
                    flag= 1;
                    break;
                }
                q = q.next;
            }
            //说明插在最后一个
            if(flag==0){
                q.next = temp;
                temp.next = null;
            }
        }
        return dummyNode.next;
    }
}

第二种,推荐面试直接写,堆排序,借助优先队列,能5分钟秒了

java 复制代码
/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode sortList(ListNode head) {
        //推荐使用优先队列
        PriorityQueue<ListNode> queue = new PriorityQueue<>(new Comparator<>(){
            public int compare(ListNode a ,ListNode b){
                return a.val-b.val;
            }
        });
        ListNode p = head;
        while(p!=null){
            queue.offer(p);
            p=p.next;
        }
        ListNode dummy = new ListNode(0);
        ListNode q = dummy;
        while(queue.isEmpty()==false){
            ListNode node = queue.poll();
            q.next = node;
            q = q.next;
            q.next = null;
        }
        return dummy.next;
    }
}

第三种,归并排序,主要就是分为归并和合并两部分。合并很简单。拆分时候,通过快慢指针拆出来两个链表头。

java 复制代码
class Solution {
    public ListNode sortList(ListNode head) {
        if(head==null||head.next==null)return head;
        ListNode slow = head;
        //这里一定一定要slow和fast分开
        //不然 4 2 会出现,head一直是4,newHead一直是空的情况
        ListNode fast = head.next;
        while(fast!=null&&fast.next!=null){
            fast = fast.next.next;
            slow = slow.next;
        }
        ListNode newHead = slow.next;
        slow.next = null;
        return merge(sortList(head),sortList(newHead));
    }
    public ListNode merge(ListNode l1,ListNode l2){
        ListNode p = l1;
        ListNode q = l2;
        ListNode dummy = new ListNode(0);
        ListNode t = dummy;
        while(p!=null&&q!=null){
            if(p.val<q.val){
                t.next = p;
                p=p.next;
            }else{
                t.next = q;
                q=q.next;
            }
            t=t.next;
        }
        t.next = p==null?q:p;
        return dummy.next;
    }
}
相关推荐
Amd7941 小时前
深入探讨索引的创建与删除:提升数据库查询效率的关键技术
数据结构·sql·数据库管理·索引·性能提升·查询优化·数据检索
OKkankan6 小时前
实现二叉树_堆
c语言·数据结构·c++·算法
指尖下的技术7 小时前
Mysql面试题----为什么B+树比B树更适合实现数据库索引
数据结构·数据库·b树·mysql
Bunury10 小时前
组件封装-List
javascript·数据结构·list
Joeysoda10 小时前
Java数据结构 (从0构建链表(LinkedList))
java·linux·开发语言·数据结构·windows·链表·1024程序员节
比特在路上10 小时前
ListOJ14:环形链表II(寻找环的入口点)
数据结构·链表
涅槃寂雨14 小时前
C语言小任务——寻找水仙花数
c语言·数据结构·算法
『往事』&白驹过隙;14 小时前
操作系统(Linux Kernel 0.11&Linux Kernel 0.12)解读整理——内核初始化(main & init)之缓冲区的管理
linux·c语言·数据结构·物联网·操作系统
就爱学编程14 小时前
从C语言看数据结构和算法:复杂度决定性能
c语言·数据结构·算法
半桔14 小时前
栈和队列(C语言)
c语言·开发语言·数据结构·c++·git