[编程之美]区间重合判断

编程之美2.19,在网上看见很多答案但是并没有看见java版本的,所以自己写一个。

问题描述:

给定一个源区间[x, y] (y>=x)和N个无序的目标区间[x1,y1],[x2,y2],[x3,y3],……[xN,yN],判断源区间[x,y]是不是在目标区间内?
例如给定源区间[1,6]和一组无序的目标区间[2,3],[1,2],[2,9],[3,4],即可认为区间[1,6]在区间[2,3],[1,2],[2,9],[3,4]内。

解法:

  1. 先对区间进行排序,这里自己定义了一个比较器,利用Arrays.sort排序。O(nlogn)
  2. 合并重合的区间,确定合并后区间的个数。O(n)
  3. 二分查找目标数组,如果在同一个区间,则说明存在重合。O(logn)

下面是代码

    static class numsComparator implements Comparator<int[]>{
        @Override
        public int compare(int[] arg0, int[] arg1) {
            return (arg0[0]<arg1[0]? -1: (arg0[0]==arg1[0]? 0: 1));
        }

    }

    private static int lineMerge(int[][] nums){
        int len=0;
        int low=nums[0][0];
        int high=nums[0][1];
        for(int i=1; i<nums.length; i++){
            if(nums[i][0]<=high){
                high=Math.max(high, nums[i][1]);
            }else{
                nums[len][0]=low;
                nums[len][1]=high;
                len++;
                low=nums[i][0];
                high=nums[i][1];
            }
        }
        nums[len][0]=low;
        nums[len][1]=high;
        return ++len;

    }

    private static int binarySearch(int[][] nums, int len, int target){
        int i=0;
        int j=len-1;
        while(i<=j){
            int mid=(i+j)>>>1;
            if(nums[mid][0]<=target&&nums[mid][1]>=target){
                return mid;
            }else if(nums[mid][1]<target){
                i=mid+1;
            }else{
                j=mid-1;
            }
        }
        return -1;      
    }   

    public static boolean linesCover(int[][] nums, int[] target){
        //step1: 按区间低位进行排序
        Arrays.sort(nums, new numsComparator() );
        System.out.println(Arrays.deepToString(nums));
        //step2: 合并区间,新的区间个数为len
        int len=lineMerge(nums);
        for(int i=0; i<len; i++){
            System.out.println(Arrays.toString(nums[i]));
        }
        //step3: 二分查找
        int i=binarySearch(nums, len, target[0]);
        int j=binarySearch(nums, len, target[1]);
        //在同一区间且不为-1
        return (i!=-1&&i==j);
    }

    public static void main(String[] args) {
        int[][] nums2={{2, 3}, {1, 2}, {3, 9}, {12, 15}, {4, 5}};
        int[] nums3={1, 6};
        System.out.println(linesCover(nums2, nums3));

    }

测试用例比较少,不知道是否有疏漏,欢迎指正。

### 判断两个时间区间是否重叠的方法 在编程中,判断两个时间区间是否重叠是一个常见的需求。以下是基于提供的引用内容和专业知识的一种解决方案。 #### 方法概述 为了判断两个时间区间 `[startTime1, endTime1]` 和 `[startTime2, endTime2]` 是否存在重合部分,可以采用以下逻辑: 如果 `endTime1 >= startTime2 && endTime2 >= startTime1` 成立,则说明这两个时间区间有交集[^2]。这是因为只有当第一个区间的结束时间晚于第二个区间的开始时间,并且第二个区间的结束时间晚于第一个区间的开始时间时,两者才会发生重叠。 #### Java 实现代码示例 下面提供了一个简单的 Java 函数用于检测两个时间区间是否存在重叠: ```java import java.time.LocalTime; public class TimeOverlapChecker { public static boolean isOverlapping(LocalTime startTime1, LocalTime endTime1, LocalTime startTime2, LocalTime endTime2) { // 如果 end1 大于等于 start2 并且 end2 大于等于 start1,则表示有重叠 return !endTime1.isBefore(startTime2) && !endTime2.isBefore(startTime1); } public static void main(String[] args) { LocalTime startTime1 = LocalTime.parse("09:00"); LocalTime endTime1 = LocalTime.parse("11:00"); LocalTime startTime2 = LocalTime.parse("10:00"); LocalTime endTime2 = LocalTime.parse("12:00"); if (isOverlapping(startTime1, endTime1, startTime2, endTime2)) { System.out.println("The time intervals overlap."); } else { System.out.println("The time intervals do not overlap."); } } } ``` 上述代码通过比较两个时间段的最大值和最小值得到结论。这里使用了 `LocalTime` 类型来处理时间数据,确保程序能够正确解析并操作 24 小时制的时间字符串。 #### C语言实现代码示例 对于C语言的情况,也可以编写类似的函数来进行判断。假设输入的是整数形式的小时分钟组合(如HHMM),则可以用如下方法: ```c #include <stdio.h> #include <stdbool.h> bool areIntervalsOverlapping(int startTime1, int endTime1, int startTime2, int endTime2){ return !(endTime1 < startTime2 || endTime2 < startTime1); } int main(){ int startTime1 = 900; int endTime1 = 1100; int startTime2 = 1000; int endTime2 = 1200; if(areIntervalsOverlapping(startTime1,endTime1,startTime2,endTime2)){ printf("The time intervals overlap.\n"); } else{ printf("The time intervals do not overlap.\n"); } return 0; } ``` 此段代码实现了基本的功能,即检查两个由整数值代表的时间段之间是否有任何交叉点[^5]。 ### 结论 无论是哪种语言环境下的应用开发,核心算法都保持一致——只需验证两端点之间的关系即可得出结论关于两段时间段是否会相互覆盖的信息。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值