Interval 间隔问题
2018-09-07 09:03:14
一、Merge Intervals
问题描述:

问题求解:
public List<Interval> merge(List<Interval> intervals) {
List<Interval> res = new ArrayList<>();
if (intervals.size() == 0) return res;
Collections.sort(intervals, new Comparator<Interval>() {
public int compare(Interval o1, Interval o2) {
return o1.start - o2.start;
}
});
int start = intervals.get(0).start;
int end = intervals.get(0).end;
for (int i = 1; i < intervals.size(); i++) {
if (intervals.get(i).start > end) {
res.add(new Interval(start, end));
start = intervals.get(i).start;
end = intervals.get(i).end;
}
else {
end = Math.max(end, intervals.get(i).end);
}
}
res.add(new Interval(start, end));
return res;
}
二、Insert Interval
问题描述:

问题求解:
本题的问题描述中明确的说明了,本题的给出条件中的intervals是已经排序好的,并且是没有overlapping的,因此在后续的求解过程中只需要一次遍历即可。
public List<Interval> insert(List<Interval> intervals, Interval newInterval) {
List<Interval> res = new ArrayList<>();
int i = 0;
while (i < intervals.size() && intervals.get(i).end < newInterval.start) {
res.add(intervals.get(i++));
}
while (i < intervals.size() && intervals.get(i).start <= newInterval.end) {
newInterval.start = Math.min(newInterval.start, intervals.get(i).start);
newInterval.end = Math.max(newInterval.end, intervals.get(i).end);
i++;
}
res.add(newInterval);
while (i < intervals.size()) res.add(intervals.get(i++));
return res;
}
三、My Calendar I
问题描述:

问题求解:
解法一:Boundary Counting
对边界进行计数,最后遍历一遍即可,如果过程中有curSum大于1的情况,则表明出现了overlapping。
如果使用keySet()则会多出log(n)的时间,而本题卡时间非常紧,如果使用key进行提取,则会TLE。
如果使用entrySet(),则会Accept,但是也是将将通过。
public class MyCalendar {
TreeMap<Integer, Integer> map;
public MyCalendar() {
map = new TreeMap<>();
}
public boolean book(int start, int end) {
return helper(start, end);
}
private boolean helper(int start, int end) {
map.put(start, map.getOrDefault(start, 0) + 1);
map.put(end, map.getOrDefault(end, 0) - 1);
int curSum = 0;
for (Map.Entry<Integer, Integer> entry : map.entrySet()) {
curSum += entry.getValue();
if (curSum > 1) {
map.put(start, map.get(start) - 1);
if (map.get(start) == 0) map.remove(start);
map.put(end, map.get(end) + 1);
if (map.get(end) == 0) map.remove(end);
return false;
}
}
return true;
}
}
解法二、
记录各个interval,并且所有的interval都是没有overlapping的。
public class MyCalendar {
TreeMap<Integer, Integer> treeMap;
public MyCalendar() {
treeMap = new TreeMap<>();
}
public boolean book(int start, int end) {
Integer floor = treeMap.floorKey(start);
if (floor != null && treeMap.get(floor) > start) return false;
Integer ceil = treeMap.ceilingKey(start);
if (ceil != null && ceil < end) return false;
treeMap.put(start, end);
return true;
}
}
四、My Calendar II
问题描述:

问题求解:
万能的Boundary Counting。
public class MyCalendarTwo {
TreeMap<Integer, Integer> map;
public MyCalendarTwo() {
map = new TreeMap<>();
}
public boolean book(int start, int end) {
map.put(start, map.getOrDefault(start, 0) + 1);
map.put(end, map.getOrDefault(end, 0) - 1);
int cnt = 0;
for (Map.Entry<Integer, Integer> entry : map.entrySet()) {
cnt += entry.getValue();
if (cnt > 2) {
map.put(start, map.get(start) - 1);
if (map.get(start) == 0) map.remove(start);
map.put(end, map.get(end) + 1);
if (map.get(end) == 0) map.remove(end);
return false;
}
}
return true;
}
}
五、My Calendar III
问题描述:

问题求解:
解法一:
万能的Boundary Counting。
public class MyCalendarThree {
TreeMap<Integer, Integer> map;
public MyCalendarThree() {
map = new TreeMap<>();
}
public int book(int start, int end) {
map.put(start, map.getOrDefault(start, 0) + 1);
map.put(end, map.getOrDefault(end, 0) - 1);
int res = 0;
int cnt = 0;
for (Map.Entry<Integer, Integer> entry : map.entrySet()) {
cnt += entry.getValue();
if (res < cnt) res = cnt;
}
return res;
}
}
解法二:
线段树求解,效率有较大的提升。
public class MyCalendarThree {
SegmentTree root;
int res;
public MyCalendarThree() {
root = new SegmentTree(0, 1000000000, 0);
res = 0;
}
public int book(int start, int end) {
add(start, end, root);
return res;
}
private void add(int start, int end, SegmentTree root) {
if (root.m != -1) {
if (start >= root.m) add(start, end, root.right);
else if (end <= root.m) add(start, end, root.left);
else {
add(start, root.m, root.left);
add(root.m, end, root.right);
}
return;
}
if (start == root.l && end == root.r) {
root.cnt++;
res = Math.max(res, root.cnt);
}
else if (start == root.l) {
root.m = end;
root.left = new SegmentTree(start, root.m, root.cnt + 1);
root.right = new SegmentTree(root.m, root.r, root.cnt);
res = Math.max(res, root.cnt + 1);
}
else if (end == root.r) {
root.m = start;
root.left = new SegmentTree(root.l, root.m, root.cnt);
root.right = new SegmentTree(root.m, root.r, root.cnt + 1);
res = Math.max(res, root.cnt + 1);
}
else {
root.m = start;
root.left = new SegmentTree(root.l, root.m, root.cnt);
root.right = new SegmentTree(root.m, root.r, root.cnt);
add(start, end, root.right);
}
}
}
class SegmentTree {
int l;
int r;
int m; // m : 分割点,如果尚未分割则为-1。
int cnt;
SegmentTree left;
SegmentTree right;
SegmentTree(int l, int r, int cnt) {
this.l = l;
this.r = r;
this.m = -1;
this.cnt = cnt;
this.left = null;
this.right = null;
}
}
六、Interval List Intersections
问题描述:


问题求解:
如何快速判断是否相交呢?

public int[][] intervalIntersection(int[][] A, int[][] B) {
List<int[]> res = new ArrayList<>();
int i = 0;
int j = 0;
while (i < A.length && j < B.length) {
int s = Math.max(A[i][0], B[j][0]);
int e = Math.min(A[i][1], B[j][1]);
if (s <= e) res.add(new int[]{s, e});
if (A[i][1] < B[j][1]) i++;
else j++;
}
int[][] rst = new int[res.size()][2];
for (i = 0; i < res.size(); i++) {
rst[i][0] = res.get(i)[0];
rst[i][1] = res.get(i)[1];
}
return rst;
}
Interval 间隔问题的更多相关文章
- iOS 性能调试
性能调优的方式: 1.通过专门的性能调优工具 2.通过代码优化 1. 性能调优工具: 下面针对iOS的性能调优工具进行一个介绍: 1.1 静态分析工具–Analyze 相信iOS开发者在App进行Bu ...
- Python任务调度模块 – APScheduler
APScheduler是一个Python定时任务框架,使用起来十分方便.提供了基于日期.固定时间间隔以及crontab类型的任务,并且可以持久化任务.并以daemon方式运行应用.目前最新版本为3.0 ...
- Visifire Chart相关属性详解
<vc:Chart x:Name="HourlyChart" Theme="Theme1" Grid.Row="1" xmlns:vc ...
- 使用MonkeyTest对Android客户端进展压力测试
Monkey是Android中的一个命令行工具,可以运行在模拟器里或实际设备中.它向系统发送伪随机的用户事件流(如按键输入.触摸屏输入.手势输入等),实现对正在开发的应用程序进行压力测试. 先来看一条 ...
- windows 环境和linux环境下 ping命令的区别:
Ping 是Windows自带的一个DOS命令.利用它可以检查网络是否能够连通,用好它可以很好地帮助我们分析判定网络故障.该命令可以加许多参数使用,键入Ping按回车即可看到详细说明.Ping 命令可 ...
- iOS-掌握了时间就掌握了一切!
Demo下载地址 一. NSDate相关知识 1.获取当前时间 [NSDate date]; 注意: 获取的当前时间是世界时间:比我们用的时间慢8个小时. 2.世界时间转化为本地时间 - (void) ...
- 数字限时增长效果实现:numberGrow.js
这是上周工作中写到的一个功能,大概的效果就是页面中有几处数字,统计公司的一些业务信息,需要在第一次出现的时候,做一个从0开始增长,大概2秒自动增长到真实数值,并停止增长的效果.这个问题的重点在于解决如 ...
- jvm指令调试
监控GC的工具分为2种:命令行工具和图形工具: 常用的命令行工具有: 注:下面的命令都在JAVA_HOME/bin中,是java自带的命令.如果您发现无法使用,请直接进入Java安装目录调用或者先设置 ...
- Python学习笔记之抽象
一.创建函数 >>> import math >>> x=1 >>> y=math.sqrt >>> callable(x) # ...
随机推荐
- 【题解】Luogu UVA12345 Dynamic len(set(a[L:R]))
原题传送门 这题要用动态莫队,我博客里有介绍 这道题和luogu P1903 [国家集训队]数颜色 / 维护队列差不多,解法就在上面那篇博客里qaq 主要的问题是如何排序? 排序有三个关键字: 1.左 ...
- VC++ 判断一个文件是不是快捷方式
转载:https://bbs.csdn.net/topics/34999 #include <iostream> #include <Shlobj.h> #include &l ...
- shell脚本之 给PNG图片添加后缀@3x
1,给png图片加上后缀@3x #!/bin/sh #root_src=$(dirname $(PWD)) #echo ${root_src} image_path=${root_src}/image ...
- maven项目更新之后,JDK版本成为1.5
描述:maven项目更新之后,JDK版本成为1.5? 解决:在pom.xml文件中配置java版本,选中build path 设置之后,刷新maven项目
- Python3 tkinter基础 Listbox Scrollbar 创建垂直滚动条
Python : 3.7.0 OS : Ubuntu 18.04.1 LTS IDE : PyCharm 2018.2.4 Conda ...
- Hunter’s Apprentice 【判断多边形边界曲线顺逆时针】
问题 H: Hunter's Apprentice 时间限制: 1 Sec 内存限制: 128 MB 提交: 353 解决: 39 [提交] [状态] [命题人:admin] 题目描述 When ...
- HDU - 1849 Rabbit and Grass 【Nim博弈】
Problem Description 大学时光是浪漫的,女生是浪漫的,圣诞更是浪漫的,但是Rabbit和Grass这两个大学女生在今年的圣诞节却表现得一点都不浪漫:不去逛商场,不去逛公园,不去和AC ...
- Nvme固体硬盘Intel750,SM961分别使用一段时间以后对比
在SM961使用了一年半(2017年1月17日购买)后,再次测试,这次测试使用AS_SSD_Benchmark工具进行测试 感觉CrystalDiskMark工具测出来的分数在所以工具中分数最高 看图 ...
- 【Spring Security】七、RememberMe配置
一.概述 RememberMe 是指用户在网站上能够在 Session 之间记住登录用户的身份的凭证,通俗的来说就是用户登陆成功认证一次之后在制定的一定时间内可以不用再输入用户名和密码进行自动登录.这 ...
- hihoCoder week20 线段树的区间修改
区间修改 区间查询 最后一场比赛前的无可救药的热身 #include <bits/stdc++.h> using namespace std; #define mid ((l+r)/2) ...