本文始发于个人公众号:TechFlow,原创不易,求个关注

今天是LeetCode专题的第33篇文章,我们一起来看LeetCode的第56题,它的难度是Medium。

题意

这道题的题意也很简单,只有一句话:“Given a collection of intervals, merge all overlapping intervals.”

interval是间隔、区间的意思,也就是说题目会给我们一系列区间,让我们把这些区间合并在一起。

我们看下题目给的样例来感受一下:

Input: [[1,3],[2,6],[8,10],[15,18]]
Output: [[1,6],[8,10],[15,18]]
Explanation: Since intervals [1,3] and [2,6] overlaps, merge them into [1,6].

Input: [[1,4],[4,5]]
Output: [[1,5]]
Explanation: Intervals [1,4] and [4,5] are considered overlapping.

分析

通过观察样例,我们发现题目通过数组给定区间,每个区间有两个端点。两个区间能够合并的条件,就是互相之间有交叉的部分。我们看下下图,这应该很直观。

当两个区间[s1, e1]和[s2, e2]中的e1 >= s2时,这两个区间就可以进行合并。合并之后得到的新区间是[s1, e2]。

但是这存在一个小问题,我们如何能判断第一个区间一定在第二个区间的左侧呢,会不会发生重叠呢?

如果是这种情况那么合并之后的结果就是[s2, e2]了,另外一个问题是,这样的区间一共有N个,我们怎么判断合并的顺序呢?很有可能出现AB两个区间原本不能合并,但是A合并了区间C之后又可以和B合并的情况。如果我们枚举的话会很麻烦,我们不但需要考虑合并的时候会发生的种种情况,还需要考虑合并的发生顺序。而且我们也很难得知是否所有能够合并的区间已经合并完成。

题解

我们梳理一下目前遇到的问题,第一个问题是区间根据位置的不同合并之后的结果可能有多个。第二个问题是区间合并之后会创建新的合并的可能,第三个问题是我们判断当前是否还有合并的可能开销很大

其中第三个问题是前两个问题导致的,只要解决了其中一个,第三个问题自然迎刃而解。其中第二个问题是无法解决的,因为这是区间合并的天然属性,我们执行区间合并必然会有这样的情况发生。所以我们只能针对第一个问题下手,合并之后的结果可能有多种的本质原因是区间的位置关系可能有多个。A和B合并,A可以出现在B的左侧,也可以出现在B的右侧。再根据区间长短关系,所以才导致了多种结果。

如果我们能够保证A一定出现在B的左侧,那么A和B就只有三种情况。一种是A和B不相交,也就是不能合并。第二种是A和B相交,第三种是A包含B。

我们把图画出来,很容易发现后面两种能够合并的情况其实可以概括成一种,它们合并之后的结果都是[s1, max(e1, e2)]。

既然如此,本题的关键就是区间之间的顺序。假如我们保证了区间的顺序,我们依次合并显然可以很容易得到结果。但是我们怎么得到区间顺序呢?

其实很简单,就是排序,既然区间本来没有顺序,我们对它们排序,不就有顺序了吗?

我们可以规定左侧端点小的区间排在前面,如果两个区间左端点一致,右端点小的靠前。这是什么?这其实就是字典序排序,在Python当中我们可以直接调用sorted对多元素的list进行字典序排序。如果是其他语言,也不麻烦,我们可以自己定义比较算子,一样可以解决。

既然区间有序了,我们只需要从左到右遍历就可以覆盖所有情况了,第三个问题也就解决了。

最后,我们来看下代码,只要想到了排序,这道题并不难。但是初学者往往很难想到排序上,这当中的思维推导过程才是我们需要熟悉的。

class Solution:
def merge(self, intervals: List[List[int]]) -> List[List[int]]:
ret = []
if len(intervals) == 0:
return ret
# 字典序排序
intervals = sorted(intervals)
# l,r表示当前区间的两个端点
l, r = intervals[0]
# 从1开始遍历区间,进行合并
for i in range(1, len(intervals)):
s, e = intervals[i]
# 如果可以合并,维护合并之后的右端点
if s <= r:
r = max(r, e)
else:
# 否则加入答案,将i区间作为当前进行合并的区间
ret.append([l, r])
l, r = s, e
    ret.append([l, r])
<span class="hljs-keyword" style="color: #333; font-weight: bold; line-height: 26px;">return</span> ret

结尾

到这里,这道题就结束了,除了排序之外,这道题还可以使用连通分量的思想来做。我们可以枚举所有区间的关系,我们可以把所有区间看成是图上的一个点。只要两个区间可以合并,我们就把它们两个点之间连一条边。这样所有可以合并在一起的区间,就构成了一个连通分量。我们最后遍历这整张图上所有的连通分量就可以拿到所有合并之后的区间结果。和排序比起来,这个算法显然复杂得多,需要建图、连边,然后计算连通分量。并且复杂度也很高,是的算法。我觉得完全没有必要,所以就不多介绍了,感兴趣的同学可以自己了解一下,只要图建好之后,连通分量的求解也不是很复杂。

你看,复杂的算法未必效果就好,还是要适合题目的才是最好的,但什么算法是适合题目的呢?这才是所有问题的关键。这不禁让我想到了电影霍元甲里面,李连杰说的一句话,武功本身没有强弱之别,只是使用武功的武者实力有高下之分。算法也是一样,复杂的算法并不一定就强,灵活运用、理性分析、合理运用才是王道。

今天的文章就到这里,原创不易,需要你一个关注的支持,你的举手之劳对我来说很重要。

LeetCode 56,区间合并问题的更多相关文章

  1. leetcode 56区间合并

    class Solution { public: static bool cmp(vector<int> a,vector<int> b){ ]<b[]; } vecto ...

  2. 【LeetCode】区间合并

    给定一组区间,将所有区间重叠的部分合并起来. e.g. 给出 { [1, 3], [2, 6], [8, 10], [15, 18] },返回 { [1, 6], [8, 10], [15, 18] ...

  3. LeetCode 56. Merge Intervals 合并区间 (C++/Java)

    题目: Given a collection of intervals, merge all overlapping intervals. Example 1: Input: [[1,3],[2,6] ...

  4. LeetCode 56,57,60,连刷三题不费劲

    本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是LeetCode专题的第34篇文章,刚好接下来的题目比较简单,很多和之前的做法类似.所以我们今天出一个合集,一口气做完接下来的57.5 ...

  5. POJ 3667 Hotel(线段树 区间合并)

    Hotel 转载自:http://www.cnblogs.com/scau20110726/archive/2013/05/07/3065418.html [题目链接]Hotel [题目类型]线段树 ...

  6. HDU 3911 线段树区间合并、异或取反操作

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=3911 线段树区间合并的题目,解释一下代码中声明数组的作用: m1是区间内连续1的最长长度,m0是区间内连续 ...

  7. HDU 1540 Tunnel Warfare 平衡树 / 线段树:单点更新,区间合并

    Tunnel Warfare                                  Time Limit: 4000/2000 MS (Java/Others)    Memory Lim ...

  8. HDU 3911 Black And White(线段树区间合并+lazy操作)

    开始以为是水题,结果...... 给你一些只有两种颜色的石头,0为白色,1为黑色. 然后两个操作: 1 l r 将[ l , r ]内的颜色取反 0 l r 计算[ l , r ]内最长连续黑色石头的 ...

  9. POJ 2750 Potted Flower (线段树区间合并)

    开始懵逼找不到解法,看了网上大牛们的题解才发现是区间合并...  给你n个数形成一个数列环,然后每次进行一个点的修改,并输出这个数列的最大区间和(注意是环,并且区间最大只有n-1个数) 其实只需要维护 ...

随机推荐

  1. javascript入门 之 zTree(十二 托拽事件(二))

    1.逻辑可能有不完善的地方,如果发现,请指出. <!DOCTYPE html> <HTML> <HEAD> <TITLE> ZTREE DEMO - d ...

  2. python 判断一个字符串是否是小数

    """练习判断一个小数1.判断是否合法2.合法需要有一个小数点3.小数点左边必须是个整数,右边必须是个正整数 """ def xiaoshu ...

  3. JS入门系列(1)-原型-函数原型

    实例1: 首先定义一个Persion类或者说是函数 var p1 = Persion();:表示,作为普通函数调用 var p2 = new Persion();:表示,作为构造器调用 创建函数之后, ...

  4. python3(二十六)slots

    """ """ __author__ = 'shaozhiqi' # python是动态语言,所以定义类后,我们可以动态的给类绑定属性和方法 ...

  5. 《深入理解 Java 虚拟机》笔记整理

    正文 一.Java 内存区域与内存溢出异常 1.运行时数据区域 程序计数器:当前线程所执行的字节码的行号指示器.线程私有. Java 虚拟机栈:Java 方法执行的内存模型.线程私有. 本地方法栈:N ...

  6. cmake添加版本号

    vVersion.cmake文件内容如下: #vversion.cmake #vDateTime string(TIMESTAMP vDateTime "%Y%m%d-%H%M%S" ...

  7. 如何用 Python 绘制玫瑰图等常见疫情图

    新冠疫情已经持续好几个月了,目前,我国疫情已经基本控制住了,而欧美国家正处于爆发期,我们会看到很多网站都提供了多种疫情统计图,今天我们使用 Python 的 pyecharts 框架来绘制一些比较常见 ...

  8. dubbo(三):负载均衡实现解析

    dubbo作为分布式远程调用框架,要保证的点很多,比如:服务注册与发现.故障转移.高性能通信.负载均衡等等! 负载均衡的目的是为了特定场景下,能够将请求合理地平分到各服务实例上,以便发挥所有机器的叠加 ...

  9. 当文件目录变得杂乱不堪怎么办,python帮你轻松搞定

    这几天和几个小伙伴,在一起合做一个ppt. 做ppt之前有原版的ppt,和一个word大纲,在制作过程中,又不断添加图片.视频等素材,最终,整个目录变得杂乱不堪(见下图-处理之前) 那我想,可不可以做 ...

  10. frame/iframe多表单切换

    应用场景: 在Web应用中经常会遇到frame/iframe表单嵌套页面的应用,WebDriver只能在一个页面上对元素识别与定位,对于frame/iframe表单内嵌页面上的元素无法直接定位.这时就 ...