56. 合并区间

56. 合并区间

中等

以数组 intervals 表示若干个区间的集合,其中单个区间为 intervals[i] = [starti, endi] 。请你合并所有重叠的区间,并返回 一个不重叠的区间数组,该数组需恰好覆盖输入中的所有区间

示例 1:

复制代码
输入:intervals = [[1,3],[2,6],[8,10],[15,18]]
输出:[[1,6],[8,10],[15,18]]
解释:区间 [1,3] 和 [2,6] 重叠, 将它们合并为 [1,6].

示例 2:

复制代码
输入:intervals = [[1,4],[4,5]]
输出:[[1,5]]
解释:区间 [1,4] 和 [4,5] 可被视为重叠区间。

提示:

  • 1 <= intervals.length <= 104
  • intervals[i].length == 2
  • 0 <= starti <= endi <= 104

📝 核心笔记:合并区间 (排序 + 贪心)

1. 核心思想 (一句话总结)

"先排序,再接龙"。

只要按照左端点排序,重叠的区间就一定会挨在一起。我们只需要看当前区间能不能"接"上结果集里的最后一个区间。

2. 算法流程 (三步走)
  1. 排序 (Sort): 必须按照 左端点 (start) 从小到大排序。
  2. 判断 (Check):
    • 如果 当前左端点 p[0]****<=结果集最后一个右端点 last[1]->撞车了/重叠了
    • 否则 ->没重叠,直接加入新区间。
  1. 合并 (Merge):
    • 发生重叠时,不要直接把当前区间的右端点赋给上一个。
    • 关键点: 要取最大值 Math.max(last[1], p[1])。因为可能存在"大包小"的情况(如 [1, 10][2, 3],合并后还是 [1, 10])。

🔍 代码回忆清单 (带注释版)

复制代码
// 题目:LC 56. 合并区间
class Solution {
    public int[][] merge(int[][] intervals) {
        // 关键点1:必须排序!按左端点升序
        // lambda 写法:(a, b) -> a[0] - b[0]
        Arrays.sort(intervals, (p, q) -> p[0] - q[0]); 

        List<int[]> ans = new ArrayList<>();
        
        for (int[] p : intervals) {
            int m = ans.size();
            // 关键点2:判断是否重叠
            // 必须 ans 不为空,且 当前左 <= 上一个右
            if (m > 0 && p[0] <= ans.get(m - 1)[1]) { 
                // 关键点3:合并动作
                // 吞并!更新右边界为两者的最大值 (千万别忘了 Math.max)
                ans.get(m - 1)[1] = Math.max(ans.get(m - 1)[1], p[1]); 
            } else {
                // 不重叠,另起炉灶
                ans.add(p); 
            }
        }
        // 关键点4:List 转 二维数组的标准写法
        return ans.toArray(new int[ans.size()][]);
    }
}

场景一:接龙(延长)------ 这是你直觉想到的情况

假设前一个区间是 [1, 5],当前区间是 [2, 8]

  • 前一个 (ans.get(m-1)): [1, ----- 5]
  • 当前 (p): [2, -------- 8]
  • 合并结果 :应该变成 [1, 8]

在这种情况下,p[1] (8) 确实比 ans.get(m-1)[1] (5) 大。 Math.max(5, 8) 选了 8 。这是符合直觉的:区间变长了


场景二:包含(大包小)------ 这是必须要 Math.max 的原因!

假设前一个区间是 [1, 10],当前区间是 [2, 6]。 (注意:因为我们按左端点排序了,所以 [2, 6] 肯定排在 [1, 10] 后面处理)。

  • 前一个 (ans.get(m-1)): [1, ---------------- 10]
  • 当前 (p): [2, ---- 6]
  • 合并结果 :应该保持 [1, 10]

如果你不用 Max,直接赋值 ans.get(m-1)[1] = p[1]: 结果就会变成 [1, 6]后果: 你把原本 610 这一段合法的范围给丢掉了!❌

使用 Max 的逻辑: Math.max(10, 6) 选了 10 。 这样才能保证:原本覆盖的范围,绝对不会因为合并了一个小区间而变短。


⚡ 快速复习 CheckList (易错点)

  • \] **忘了排序?** 不排序这道题做不出来,必须按 start 排序。

    • 反例:[1, 5][2, 3]。如果不取 max,直接赋值变成 [1, 3] 就错了,应该是 [1, 5]
  • \] **返回值写法?** Java 中 List 转二维数组比较啰嗦:`list.toArray(new int[list.size()][])`。

想象你在安排会议室

  1. 先把所有预定单按开始时间排好序。
  2. 你拿着第一张单子 9:00 - 10:00 贴在门上。
  3. 第二张单子 9:30 - 11:00 来了。
  4. 一看 9:3010:00 之前(重叠了),于是你把门上的单子改成 9:00 - 11:00(取结束时间晚的那个)。
  5. 第三张单子 12:00 - 13:00 来了。
  6. 12:0011:00 之后(没重叠),你在门上贴第二张单子。
相关推荐
Python私教2 分钟前
如意Agent日志系统重构:从 print() 大海捞针到结构化可观测性栈
java·前端·重构
迷途之人不知返4 分钟前
优先级队列:priority_queue
数据结构·c++
jieyucx8 分钟前
Go 零基础数据结构:顺序表(像「排抽屉」一样学增删改查)
java·数据结构·golang
曦夜日长8 分钟前
C++ STL容器string(一):string的变量细节、默认函数的认识以及常用接口的使用
java·开发语言·c++
想唱rap14 分钟前
应用层协议与序列化
linux·运维·服务器·网络·数据结构·c++·算法
北山有鸟15 分钟前
IS_ERR 判断出错后,再用 PTR_ERR 把它强制转换回 int 型的错误码作为函数的返回值。
java·开发语言
程序员老邢19 分钟前
【产品底稿 11】架构规整收官:从混乱到清晰,工程结构、表命名、模块分层一次性定型
后端·架构·springboot·产品底稿·架构规整·模块分层·数据库规范
重生之我是Java开发战士19 分钟前
【笔试强训】Week3:重排字符串,分组,DNA序列
算法
We་ct21 分钟前
LeetCode 97. 交错字符串:动态规划详解
前端·算法·leetcode·typescript·动态规划
phltxy21 分钟前
深度解析:Spring Cloud Gateway 从入门到实战
java·开发语言