zoukankan      html  css  js  c++  java
  • 缘来是区间重叠呀?

    最近的一个活动配置需求,有一个非常奇葩的配置字段,可以配置活动在每天的某些时间段上线,如下图中“每日限时”,尽管已经是傻瓜操作了,但是考虑到严谨性,(实际上是为了防止产品手抖),如果两个时间段之间有重叠,应该合并两个时间段,比如下图两个时间段按道理应该合并成20:49~22:49

    那么从代码层面该如何合并这两个时间段呢?首先需要判断两个时间段是否重叠,两个时间段如果有重叠的话,从正面考虑的话,它们的关系无外乎下面四种情况。


    StartA <= EndB && StartB <= EndA
    也就是说如果满足下述条件,则RangeA和RangeB有重叠

    (StartA <= StartB && EndA >= StartB) || (StartB <= StartA && EndB >= StartA)
    

    从反面考虑的话,似乎更容易理解,如下图,考虑时间段不重叠的情况。

    也就是说下面的条件下,两个时间不会发生重叠

    EndA < StartB || StartA > EndB
    

    对上面的条件取反就是两个时间段会重叠。根据摩根定律

    !(EndA < StartB || StartA > EndB) = (EndA >= StartB) && (StartA <=EndB)
    

    如果能判断出两个时间段有重叠部分,合并两个时间段非常简单:

    Start = Max.min(StartA, StartB)
    End = Max.max(EndA, EndB)
    

    对于业务中遇到的时间段重叠问题,直接比较时间大小比较困难,只需要把所有时间转换成时间戳比较就简单的多了

    StartA = new Date(StartA).getTime()
    

    这样上述的比较问题就变成了整数之间的比较。类似的这种区间重叠问题是一个非常经典的问题,一看到这种模型,我就猜想LeetCode是否会有这种算法题。果不其然,我在LeetCode发现了完全一致的问题。解题思路很简单

    Merge Intervals

    Given a collection of intervals, merge all overlapping intervals.
    Example 1:
    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].

    /**
     * @param {number[][]} intervals
     * @return {number[][]}
     */
    var merge = function(intervals) {
      if (intervals.length < 2) {
        return intervals
      }
      // 先按照起始时间排序
      intervals.sort((v1, v2) => {
        return v1[0] - v2[0]
      })
      const result = []
      let [preStart, preEnd] = intervals[0]
      for (let len = intervals.length, i = 1; i < len; i++) {
        const [curStart, curEnd] = intervals[i]
        if (curStart <= preEnd) {
          // preEnd = Math.max(curEnd, preEnd)
          if (curEnd > preEnd) {
            preEnd = curEnd
          }
        } else {
          result.push([preStart, preEnd])
          preStart = curStart
          preEnd = curEnd
        }
      }
      result.push([preStart, preEnd])
      return result
    }
    

    另一个比较有趣的问题是,如果两个时间段有重叠部分,如何快速计算重叠部分的长度

    前面我们提到了两个区段有重叠的四种情况,为了计算两个区段的重叠部分长度,最简单的方法是先判读两个区间段属于上图的哪一种情况,然后就很容易计算出来了,比如上面的四种情况对应的重叠部分分别是:EndA - StartBEndB - StartBEndB - StartAEndA - StartA.而实际上并不需要区分上面的四种情况,上述四个计算式的最小值就能概括四种情况,即,如果两个区间段有重叠的话,其重叠部分的长度为

    Math.min(EndA - StartB,EndB - StartB,EndB - StartA,EndA - StartA)
    

    参考
    Determine Whether Two Date Ranges Overlap

  • 相关阅读:
    Android进程生命周期与ADJ
    四大组件之综述
    Linux进程pid分配法
    Linux的进程管理
    Linux硬盘管理
    Linux用户管理
    Linux命令行使用
    vim技巧5 常用操作
    vim技巧4 删除/保留文本中匹配行
    如何绘制UML图?
  • 原文地址:https://www.cnblogs.com/star91/p/yuan-lai-shi-qu-jian-zhong-die-ya.html
Copyright © 2011-2022 走看看