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

LeetCode - 搜索插入位置 / 排序链表

  ​欢迎光临小站:致橡树

搜索插入位置

给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。

请必须使用时间复杂度为 O(log n) 的算法。

示例 1:

输入: nums = [1,3,5,6], target = 5
输出: 2

示例 2:

输入: nums = [1,3,5,6], target = 2
输出: 1

示例 3:

输入: nums = [1,3,5,6], target = 7
输出: 4

提示:

  • 1 <= nums.length <= 104

  • -104 <= nums[i] <= 104

  • nums 为 无重复元素 的 升序 排列数组

  • -104 <= target <= 104

解题思路

这段代码实现了合并两个有序链表的功能,采用递归方式实现。

  1. 创建虚拟头节点

    ListNode head = new ListNode(-1);  // 创建一个值为-1的虚拟头节点
  2. 递归合并过程

    • 基线条件:当两个链表都为空时,递归结束

      if (list1 == null && list2 == null) {return head;
      }
    • 比较并连接节点

      • 当两个链表都不为空时,比较节点值,取较小者

        if (list1.val > list2.val) {head.next = new ListNode(list2.val);  // 创建新节点list2 = list2.next;  // 移动指针
        } else {head.next = new ListNode(list1.val);list1 = list1.next;
        }
      • 当一个链表为空时,直接连接另一个链表的节点

        else if (list1 == null) {head.next = new ListNode(list2.val);list2 = list2.next;
        } else if (list2 == null) {head.next = new ListNode(list1.val);list1 = list1.next;
        }
  3. 递归调用

    return doMergeTwoLists(head.next, list1, list2);  // 处理下一个节点
  4. 返回结果

    return head.next;  // 跳过虚拟头节点,返回真正的头节点
class Solution {public ListNode mergeTwoLists(ListNode list1, ListNode list2) {ListNode head = new ListNode(-1);doMergeTwoLists( head,  list1,  list2);return head.next;}public ListNode doMergeTwoLists(ListNode head, ListNode list1, ListNode list2) {if (list1 == null && list2 == null) {return head;}if (list1 != null && list2 != null) {if (list1.val > list2.val) {head.next = new ListNode(list2.val);list2 = list2.next;} else {head.next = new ListNode(list1.val);list1 = list1.next;}}else if (list1 == null) {head.next = new ListNode(list2.val);list2 = list2.next;}else if (list2 == null) {head.next = new ListNode(list1.val);list1 = list1.next;}return doMergeTwoLists( head.next,  list1,  list2);}
}

排序链表

给你链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表

示例 1:

输入:head = [4,2,1,3]
输出:[1,2,3,4]

示例 2:

输入:head = [-1,5,3,4,0]
输出:[-1,0,3,4,5]

示例 3:

输入:head = []
输出:[]

提示:

  • 链表中节点的数目在范围 [0, 5 * 104] 内

  • -105 <= Node.val <= 105

解题思路

整体算法流程

  1. 分割(Split):递归地将链表分成两半

  2. 排序(Sort):对每一半递归地进行排序

  3. 合并(Merge):将两个已排序的子链表合并成一个有序链表

方法分解说明

sortList(ListNode head)
public ListNode sortList(ListNode head) {return doSortList(head, null); // 初始调用,tail设为null表示链表末尾
}
  • 入口方法:启动排序过程

  • 参数:只需要传入链表头节点

  • 返回值:排序后的链表头节点

doSortList(ListNode head, ListNode tail)
public ListNode doSortList(ListNode head, ListNode tail) {// 基线条件:当子链表只有一个节点时返回if (head.next == tail) {head.next = null; // 断开连接,形成独立节点return head;}// 快慢指针找中点ListNode slow = head;ListNode fast = head;while (fast != tail && fast.next != tail) {slow = slow.next;fast = fast.next.next;}ListNode mid = slow;// 递归排序左右两部分ListNode left = doSortList(head, mid);ListNode right = doSortList(mid, tail);// 合并已排序的两部分return doMerge(left, right);
}
  • 分割策略

    • 使用快慢指针找到链表中点

    • 快指针每次走两步,慢指针每次走一步

    • 当快指针到达尾部时,慢指针正好在中点

  • 递归处理

    • 对左半部分(head到mid)递归排序

    • 对右半部分(mid到tail)递归排序

doMerge(ListNode left, ListNode right)
public ListNode doMerge(ListNode left, ListNode right) {ListNode dummy = new ListNode(-1); // 虚拟头节点ListNode temp = dummy;// 比较两个链表的节点值,选择较小的接入结果链表while (left != null && right != null) {if (left.val > right.val) {temp.next = right;right = right.next;} else {temp.next = left;left = left.next;}temp = temp.next;}// 处理剩余节点while (left != null) {temp.next = left;temp = temp.next;left = left.next;}while (right != null) {temp.next = right;temp = temp.next;right = right.next;}return dummy.next; // 返回合并后的链表头
}
  • 合并策略

    • 创建虚拟头节点简化操作

    • 比较两个链表的当前节点值

    • 将较小值的节点连接到结果链表

    • 移动相应链表的指针

  • 剩余处理

    • 当其中一个链表遍历完后

    • 直接将另一个链表的剩余部分连接到结果链表

关键点解析

快慢指针找中点
while (fast != tail && fast.next != tail) {slow = slow.next;fast = fast.next.next;
}
  • 终止条件

    • fast == tail:处理奇数长度链表

    • fast.next == tail:处理偶数长度链表

  • 结果

    • slow指针最终指向中点

递归终止条件
if (head.next == tail) {head.next = null; // 重要:断开原有连接return head;
}
  • 当子链表只有一个节点时终止递归

  • 必须断开next指针,否则会影响合并过程

虚拟头节点技巧
ListNode dummy = new ListNode(-1);
ListNode temp = dummy;
// ...
return dummy.next;
  • 简化链表操作

  • 避免处理头节点的特殊情况

完整代码

class Solution {public ListNode sortList(ListNode head) {return doSortList(head, null);}public ListNode doSortList(ListNode head, ListNode tail) {if (head.next == tail) {head.next = null;return head;}ListNode slow = head;ListNode fast = head;while (fast != tail && fast.next != tail) {slow = slow.next;fast = fast.next.next;}ListNode mid = slow;ListNode left = doSortList(head, mid);ListNode right = doSortList(mid, tail);return doMerge(left, right);}public ListNode doMerge(ListNode left, ListNode right) {ListNode dummy = new ListNode(-1);ListNode temp = dummy;while (left != null && right != null) {if (left.val > right.val) {temp.next = right;right = right.next;} else {temp.next = left;left = left.next;}temp = temp.next;}while (left != null) {temp.next = left;temp = temp.next;left = left.next;}while (right != null) {temp.next = right;temp = temp.next;right = right.next;}return dummy.next;}
}

http://www.lryc.cn/news/616331.html

相关文章:

  • 音视频学习(五十一):AAC编码器
  • 力扣(买卖股票的最佳时机I/II)
  • 面对信号在时频平面打结,VNCMD分割算法深度解密
  • windows的cmd命令【持续更新】
  • 数据库面试题集
  • ADB简介
  • 全面了解机器语言之kmeans
  • UE5多人MOBA+GAS 41、制作一个飞弹,添加准心索敌
  • 【走进Docker的世界】Docker环境搭建
  • Java集合框架、Collection体系的单列集合
  • OpenStack热迁移一直处于迁移中怎么办
  • Dify 从入门到精通(第 26/100 篇):Dify 的知识图谱集成
  • 基于django的宠物用品购物商城的设计与实现
  • Java基础编程核心案例:从逻辑到应用
  • Python 的列表 list 和元组 tuple 有啥本质区别?啥时候用谁更合适?
  • 嵌入式第二十四课!!linux应用软件编程与文件操作!!!
  • Java开源代码源码研究:我的成长之路与实战心得分享
  • actuary notes[2]
  • 产品经理入门 - 产品解决方案(需求分析、 功能优先级划分、功能价值、用户体验)
  • 智慧社区--4
  • Spring之【详解AOP】
  • Linux入门指南:26个基础命令全解析
  • 数字图像处理3
  • Docker-04:CGroups资源控制组
  • 【代码随想录day 15】 力扣 404. 左叶子之和
  • 部署一个免费开源的博客系统
  • OpenAI正式发布GPT-5:迈向AGI的关键一步
  • 【走进Docker的世界】深入理解Docker网络:从模式选择到实战配置
  • TF-IDF提取关键词(附实战案例)
  • 【RocketMQ 生产者和消费者】- ConsumeMessageConcurrentlyService 并发消费消息