leetcode352数据流拆分

题目信息

给你一个由非负整数 a1, a2, ..., an 组成的数据流输入,请你将到目前为止看到的数字总结为不相交的区间列表。
实现 SummaryRanges 类:
SummaryRanges() 使用一个空数据流初始化对象。
void addNum(int val) 向数据流中加入整数 val 。
int[][] getIntervals() 以不相交区间 [starti, endi] 的列表形式返回对数据流中整数的总结。
输入:
["SummaryRanges", "addNum", "getIntervals", "addNum", "getIntervals", "addNum", "getIntervals", "addNum", "getIntervals", "addNum", "getIntervals"]
[[], [1], [], [3], [], [7], [], [2], [], [6], []]
输出:
[null, null, [[1, 1]], null, [[1, 1], [3, 3]], null, [[1, 1], [3, 3], [7, 7]], null, [[1, 3], [7, 7]], null, [[1, 3], [6, 7]]]

解释:
SummaryRanges summaryRanges = new SummaryRanges();
summaryRanges.addNum(1);      // arr = [1]
summaryRanges.getIntervals(); // 返回 [[1, 1]]
summaryRanges.addNum(3);      // arr = [1, 3]
summaryRanges.getIntervals(); // 返回 [[1, 1], [3, 3]]
summaryRanges.addNum(7);      // arr = [1, 3, 7]
summaryRanges.getIntervals(); // 返回 [[1, 1], [3, 3], [7, 7]]
summaryRanges.addNum(2);      // arr = [1, 2, 3, 7]
summaryRanges.getIntervals(); // 返回 [[1, 3], [7, 7]]
summaryRanges.addNum(6);      // arr = [1, 2, 3, 6, 7]
summaryRanges.getIntervals(); // 返回 [[1, 3], [6, 7]]

题目理解

不相交区间,说明相邻的区间需要合并,即1,2应该合并为一个区间[1,2]

方法一:双指针:快慢指针

在这里插入图片描述

class SummaryRanges {
    static List<Integer> list;
    
    public SummaryRanges() {
        list = new ArrayList<>();
    }
    
    public void addNum(int val) {
        list.add(val);
    }

    /**
     *快慢指针
     * @return
     */
    public int[][] getIntervals() {
        list.sort(Comparator.comparingInt(a -> a));
        List<int[]> res = new ArrayList<>();
        int pre = -1;
        int cur = -1;
        for (Integer x : list) {
        	//pre + 1 != x 判断是否连续,即1,2,3这种情况
        	//pre != x 判断是否重复,即1,1,1这种情况
        	//pre != -1 是为了将pre赋值
            if (pre != -1 && pre + 1 != x  && pre != x) {
                res.add(new int[]{cur, pre});
                cur = x;
            }
            if (cur == -1) {
                cur = x;
            }
            pre = x;
        }
        //将最后情况添加进去
        res.add(new int[]{cur, pre});
        return res.toArray(new int[0][]);
    }
}
方法二,treeMap

方法一,每次都要排序,在遍历,有没有方法能够避免排序?方法就是在插入时实现排序,使用treeMap记录排序区间信息,转换为图论问题即可。

情况一:

[l,r]完全包裹val,则直接跳过
在这里插入图片描述

情况二:

[l,r]区间满足 r + 1 = val,则删除原区间,并增加新区间[l,val]
在这里插入图片描述

情况三:

[l,r]区间满足 l - 1 = val,则删除原区间,并增加新区间[val,r]
在这里插入图片描述

情况四:

[l0,r0],[l1,r1]两个区间满足r0 + 1 = l1 - 1 = val,则删除原来两个区间,增加新区间[l0,r1],如[1,2],[4,5],val = 3
在这里插入图片描述

情况五:

不满足以上四种情况,增加区间[val,val]
在这里插入图片描述

class SummaryRanges {
    TreeMap<Integer, Integer> preMap;

    public SummaryRanges() {
        preMap = new TreeMap<>();
    }

    public void addNum(int val) {
        //寻找l1 > val大的区间,且l1最小
        Map.Entry<Integer, Integer> entry1 = preMap.ceilingEntry(val + 1);
        //寻找l0 <= val的区间且l0最大
        Map.Entry<Integer, Integer> entry0 = preMap.floorEntry(val);
        if (entry0 != null && entry0.getKey() <= val && val <= entry0.getValue()) {
        } else {
            boolean leftFlag = entry1 != null && entry1.getKey() - 1 == val;
            boolean rightFlag = entry0 != null && entry0.getValue() + 1 == val;
            if (leftFlag && rightFlag) {
                int right = entry1.getValue();
                int left = entry0.getKey();
                preMap.remove(entry0.getKey());
                preMap.remove(entry1.getKey());
                preMap.put(left, right);
            }else if (leftFlag) {
                int right = entry1.getValue();
                preMap.remove(entry1.getKey());
                preMap.put(val, right);
            }else if (rightFlag) {
                preMap.put(entry0.getKey(), val);
            } else {
                preMap.put(val, val);
            }
        }
    }

    public int[][] getIntervals() {
        int size = preMap.size();
        int[][] ans = new int[size][2];
        int index = 0;
        for (Map.Entry<Integer, Integer> entry : preMap.entrySet()) {
            ans[index][0] = entry.getKey();
            ans[index][1] = entry.getValue();
            index++;
        }
        return ans;
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值