java---区间合并算法(每日一道算法2022.7.31)

题目
给定 n 个区间 [li,ri],要求合并所有有交集的区间。
注意如果在端点处相交,也算有交集。
输出合并完成后的区间个数。
例如:[1,3] 和 [2,6] 可以合并为一个区间 [1,6]。

输入格式
第一行包含整数 n。
接下来 n 行,每行包含两个整数 l 和 r。

输出格式
共一行,包含一个整数,表示合并区间完成后的区间个数

数据范围
1 ≤ n ≤ 100000,
−10e9 ≤ li ≤ ri≤ 10e9

输入
区间的个数
5
区间数据
1 2
2 4
5 6
7 8
7 9

输出
3
public class 区间和并 {
    public static void main(String[] args) {
        //传入数据
        Scanner in = new Scanner(System.in);
        int n = in.nextInt();
        ArrayList<link> info_dict = new ArrayList<>();
        for (int i = 0; i<n; i++) {
            info_dict.add(new link(in.nextInt(), in.nextInt()));
        }

        System.out.println(merge(info_dict));
    }
    public static int merge(ArrayList<link> info_dict) {
        //首先对info_dict进行排序,排序规则为根据link的left排序,如果left相等,根据right排序
        Collections.sort(info_dict, new Comparator<link>() {
            @Override
            public int compare(link o1, link o2) {
                int a = o1.left - o2.left;
                int b = a==0?o1.right-o2.right:a;
                return b;
            }
        });

        //创建一个用于接受合并后结果的dict,同时设置一个类似滑动窗口一样的start和end
        ArrayList<link> result_dict = new ArrayList<>();
        double start = -2e9, end = -2e9;
        for (link i : info_dict) {
            //判断两种情况
            //第一种是当end小于i.left时,说明我们遇到了一个新的没有交集的区间
            //那么就将start和end移动过来,并且将它添加到result_dict中
            if (end < i.left) {
                if (start!=-2e9) {
                    result_dict.add(new link((int)start, (int)end));
                }
                start = i.left;
                end = i.right;
            }
            //第二种情况是当有交集时,如果在当前滑动区间内,不变,如果其right大于当前end,将滑动区间的end设置为right
            else {
                end = Math.max(end, i.right);
            }
        }

        //排空滑动区间
        if (start!=-2e9) {
            result_dict.add(new link((int)start, (int)end));
        }

        //返回result_dict中区间的个数
        return result_dict.size();
    }
}

//这里的link是模仿c中的pair数据结构,java中好像没有,map的key是唯一的,所以用不了
class link {
    public int left;
    public int right;
    public link(int l, int r) {
        this.left = l;
        this.right = r;
    }
}

图解:
请添加图片描述

声明:
算法思路来源为y总,详细请见https://www.acwing.com/
本文仅用作学习记录和交流

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值