返回介绍

solution / 0300-0399 / 0352.Data Stream as Disjoint Intervals / README

发布于 2024-06-17 01:04:01 字数 5508 浏览 0 评论 0 收藏 0

352. 将数据流变为多个不相交区间

English Version

题目描述

 给你一个由非负整数 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]]

 

提示:

  • 0 <= val <= 104
  • 最多调用 addNumgetIntervals 方法 3 * 104

 

进阶:如果存在大量合并,并且与数据流的大小相比,不相交区间的数量很小,该怎么办?

解法

方法一

from sortedcontainers import SortedDict


class SummaryRanges:
  def __init__(self):
    self.mp = SortedDict()

  def addNum(self, val: int) -> None:
    n = len(self.mp)
    ridx = self.mp.bisect_right(val)
    lidx = n if ridx == 0 else ridx - 1
    keys = self.mp.keys()
    values = self.mp.values()
    if (
      lidx != n
      and ridx != n
      and values[lidx][1] + 1 == val
      and values[ridx][0] - 1 == val
    ):
      self.mp[keys[lidx]][1] = self.mp[keys[ridx]][1]
      self.mp.pop(keys[ridx])
    elif lidx != n and val <= values[lidx][1] + 1:
      self.mp[keys[lidx]][1] = max(val, self.mp[keys[lidx]][1])
    elif ridx != n and val >= values[ridx][0] - 1:
      self.mp[keys[ridx]][0] = min(val, self.mp[keys[ridx]][0])
    else:
      self.mp[val] = [val, val]

  def getIntervals(self) -> List[List[int]]:
    return list(self.mp.values())


# # Your SummaryRanges object will be instantiated and called as such:
# # obj = SummaryRanges()
# # obj.addNum(val)
# # param_2 = obj.getIntervals()
class SummaryRanges {
  private TreeMap<Integer, int[]> mp;

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

  public void addNum(int val) {
    Integer l = mp.floorKey(val);
    Integer r = mp.ceilingKey(val);
    if (l != null && r != null && mp.get(l)[1] + 1 == val && mp.get(r)[0] - 1 == val) {
      mp.get(l)[1] = mp.get(r)[1];
      mp.remove(r);
    } else if (l != null && val <= mp.get(l)[1] + 1) {
      mp.get(l)[1] = Math.max(val, mp.get(l)[1]);
    } else if (r != null && val >= mp.get(r)[0] - 1) {
      mp.get(r)[0] = Math.min(val, mp.get(r)[0]);
    } else {
      mp.put(val, new int[] {val, val});
    }
  }

  public int[][] getIntervals() {
    int[][] res = new int[mp.size()][2];
    int i = 0;
    for (int[] range : mp.values()) {
      res[i++] = range;
    }
    return res;
  }
}

/**
 * Your SummaryRanges object will be instantiated and called as such:
 * SummaryRanges obj = new SummaryRanges();
 * obj.addNum(val);
 * int[][] param_2 = obj.getIntervals();
 */
class SummaryRanges {
private:
  map<int, vector<int>> mp;

public:
  SummaryRanges() {
  }

  void addNum(int val) {
    auto r = mp.upper_bound(val);
    auto l = r == mp.begin() ? mp.end() : prev(r);
    if (l != mp.end() && r != mp.end() && l->second[1] + 1 == val && r->second[0] - 1 == val) {
      l->second[1] = r->second[1];
      mp.erase(r);
    } else if (l != mp.end() && val <= l->second[1] + 1)
      l->second[1] = max(val, l->second[1]);
    else if (r != mp.end() && val >= r->second[0] - 1)
      r->second[0] = min(val, r->second[0]);
    else
      mp[val] = {val, val};
  }

  vector<vector<int>> getIntervals() {
    vector<vector<int>> res;
    for (auto& range : mp) res.push_back(range.second);
    return res;
  }
};

/**
 * Your SummaryRanges object will be instantiated and called as such:
 * SummaryRanges* obj = new SummaryRanges();
 * obj->addNum(val);
 * vector<vector<int>> param_2 = obj->getIntervals();
 */

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文