当前位置: 首页 > article >正文

Leetcode面试经典150题-148.排序链表

题目比较简单,使用链表的归并排序

解法都在代码里,不懂就留言或者私信

合并链表部分没怎么加注释,时间实在是不充裕,看不懂的看一下这篇专门讲解合并链表的

Leetcode面试经典150题-21.合并两个有序链表-CSDN博客

/**
 * 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 {
    /**本题采用归并排序的思路:把链表分为左右两个部分,分别进行归并排序
    然后进行两个部分的merge过程,这个过程就是说合并两个有序链表的过程 */
    public ListNode sortList(ListNode head) {
        ListNode cur = head;
        /**找一下tail,这是最容易想到的,反正也就是O(n)的时间复杂度 */
        while(cur != null && cur.next != null) {
            cur = cur.next;
        }
        /**cur是最后一个节点,也就是tail */
        return mergeSortList(head, cur);
    }

    /**归并排序head~tail这个区间*/
    public ListNode mergeSortList(ListNode head, ListNode tail) {
        /**如果区间就一个节点,直接返回 */
        if(head == tail) {
            return head;
        }
        /**如果区间有两个节点,直接mergeList*/
        if(head.next == tail) {
            /**merge之前断开链接,不然就跑不完了 */
           head.next = null;
           return mergeList(head, tail);
        }
        /**开始使用快慢指针找中点 */
        ListNode slow = head, fast = head;
        /**确保fast在head~tail范围内,如果它是tail或者它的next是tail就终止循环 */
        while(fast != tail && fast.next != tail) {
            /**fast和slow先各走一步 */
            slow = slow.next;
            /**快指针每次走两步 */
            fast = fast.next.next;
        }
        /**出循环的时候fast是tail或者tail的前一个,然后slow是上班段最后一个节点,断开slow和后面链表的连接
        断开之前先拿到后半段的最后一个节点*/
        ListNode rightFirst = slow.next;
        slow.next = null;
        /**归并排序前半段 */
        ListNode list1 = mergeSortList(head, slow);
        /**不要管fast,终点是tail*/
        ListNode list2 = mergeSortList(rightFirst, tail);
        /**合并前后两个有序链表*/
        ListNode merged = mergeList(list1, list2);
        return merged;
    }
    /**合并两个有序链表的解法:除了这种方式也可以前面加个dummyNode,这样最后返回dummyNode下一个就行了 */
    public ListNode mergeList(ListNode list1, ListNode list2) {
        if(list1 == null) return list2;
        if(list2 == null) return list1;
        ListNode head = list1.val <= list2.val? list1 : list2;
        list1 = list1 == head? list1.next : list1;
        list2 = list2 == head? list2.next : list2;
        ListNode last = head;
        while(list1 != null && list2 != null) {
            if(list1.val <= list2.val) {
                last.next = list1;
                last = list1;
                list1 = list1.next;
            } else {
                last.next = list2;
                last = list2;
                list2 = list2.next;
            }
        }
        /**退出上面的while循环要么是list1用完了,要么是list2用完了,如果某个链表没有用完,就直接串到最后*/
        if(list1 != null) {
            last.next = list1;
        }
        if(list2 != null) {
            last.next = list2;
        }
        return head;
    }
}

没有特别好的解法,能过就行吧,时间复杂度都一样


http://www.kler.cn/a/306031.html

相关文章:

  • ant design 学习记录 type PrefixType = keyof typeof MOCK_DATA;
  • mac中idea菜单工具栏没有git图标了
  • Python——day09
  • Java重要面试名词整理(四):并发编程(下)
  • VSCode 性能优化指南:提高编码效率,减少资源占用
  • SDMTSP:黑翅鸢算法(Black-winged kite algorithm,BKA)求解单仓库多旅行商问题,可以更改数据集和起点(MATLAB代码)
  • 16. 池化层的基本使用 -- nn.MaxPool2d
  • 【AcWing】【Go】789. 数的范围
  • Leetcode面试经典150题-82.删除排序链表中的重复元素II
  • NISP 一级 | 5.3 电子邮件安全
  • LottieCompositionFactory.fromUrl 加载lottie的json文件
  • 《微信小程序实战(1)· 开篇示例 》
  • Python——俄罗斯方块
  • .NET/C#⾯试题汇总系列:多线程
  • 【有啥问啥】自动提示词工程(Automatic Prompt Engineering, APE):深入解析与技术应用
  • Spring security 动态权限管理(基于数据库)
  • 多源BFS的模板以及练习题(多源BFS)
  • `character_set_server` 和 `collation_server`
  • nvm安装并配置全局缓存文件
  • 【webpack4系列】webpack初识与构建工具发展(一)
  • 【GO语言】Go语言详解与应用场景分析,与Java的对比及优缺点
  • CSP组T1怪物
  • 升级VMware
  • 视频监控摄像头国标GB28181配置参数逐条解析
  • UE5安卓项目打包安装
  • Rust 控制流