[LeetCode] My Calendar I 我的日历之一
Implement a MyCalendar class to store your events. A new event can be added if adding the event will not cause a double booking.
Your class will have the method, book(int start, int end). Formally, this represents a booking on the half open interval [start, end), the range of real numbers x such that start <= x < end.
A double booking happens when two events have some non-empty intersection (ie., there is some time that is common to both events.)
For each call to the method MyCalendar.book, return true if the event can be added to the calendar successfully without causing a double booking. Otherwise, return false and do not add the event to the calendar.
Your class will be called like this: MyCalendar cal = new MyCalendar(); MyCalendar.book(start, end)
Example 1:
MyCalendar();
MyCalendar.book(10, 20); // returns true
MyCalendar.book(15, 25); // returns false
MyCalendar.book(20, 30); // returns true
Explanation:
The first event can be booked. The second can't because time 15 is already booked by another event.
The third event can be booked, as the first event takes every time less than 20, but not including 20.
Note:
- The number of calls to MyCalendar.bookper test case will be at most1000.
- In calls to MyCalendar.book(start, end),startandendare integers in the range[0, 10^9].
这道题让我们设计一个我的日历类,里面有一个book函数,需要给定一个起始时间和结束时间,与Google Calendar不同的是,我们的事件事件上不能重叠,实际上这道题的本质就是检查区间是否重叠。那么我们可以暴力搜索,对于每一个将要加入的区间,我们都和已经已经存在的区间进行比较,看是否有重复。而新加入的区间和当前区间产生重复的情况有两种,一种是新加入区间的前半段重复,并且,另一种是新加入区间的后半段重复。比如当前区间如果是[3, 8),那么第一种情况下新加入区间就是[6, 9),那么触发条件就是当前区间的起始时间小于等于新加入区间的起始时间,并且结束时间大于新加入区间的结束时间。第二种情况下新加入区间就是[2,5),那么触发条件就是当前区间的起始时间大于等于新加入区间的起始时间,并且起始时间小于新加入区间的结束时间。这两种情况均返回false,否则就将新区间加入数组,并返回true即可,参见代码如下:
解法一:
class MyCalendar {
public:
    MyCalendar() {}
    bool book(int start, int end) {
        for (auto a : cal) {
            if (a.first <= start && a.second > start) return false;
            if (a.first >= start && a.first < end) return false;
        }
        cal.push_back({start, end});
        return true;
    }
private:
    vector<pair<int, int>> cal;
};
下面这种方法将上面方法的两个if判断融合成为了一个,我们来观察两个区间的起始和结束位置的关系发现,如果两个区间的起始时间中的较大值小于结束区间的较小值,那么就有重合,返回false。比如 [3, 8) 和 [6, 9),3和6中的较大值6,小于8和9中的较小值8,有重叠。再比如[3, 8) 和 [2, 5),3和2中的较大值3,就小于8和5中的较小值5,有重叠。而对于[3, 8) 和 [9, 10),3和9中的较大值9,不小于8和10中的较小值8,所以没有重叠,参见代码如下:
解法二:
class MyCalendar {
public:
    MyCalendar() {}
    bool book(int start, int end) {
        for (auto a : cal) {
            if (max(a.first, start) < min(a.second, end)) return false;
        }
        cal.push_back({start, end});
        return true;
    }
private:
    vector<pair<int, int>> cal;
};
上面两种解法都是线性搜索,我们起始可以优化搜索时间,如果我们的区间是有序的话。所以我们用一个map来建立起始时间和结束时间的映射,map会按照起始时间进行自动排序。然后对于新进来的区间,我们在已有区间中查找第一个不小于新入区间的起始时间的区间,如果这个区间存在的话,说明新入区间的起始时间小于等于当前区间,也就是解法一中的第二个if情况,当前区间起始时间小于新入区间结束时间的话返回false。我们还要跟前面一个区间进行查重叠操作,那么判断如果当前区间不是第一个区间的话,就找到前一个区间,此时是解法一中第一个if情况,并且如果前一个区间的结束时间大于新入区间的起始时间的话,返回false。否则就建立新的映射,返回true即可,参见代码如下:
解法三:
class MyCalendar {
public:
    MyCalendar() {}
    bool book(int start, int end) {
        auto it = cal.lower_bound(start);
        if (it != cal.end() && it->first < end) return false;
        if (it != cal.begin() && prev(it)->second > start) return false;
        cal[start] = end;
        return true;
    }
private:
    map<int, int> cal;
};
参考资料:
https://discuss.leetcode.com/topic/111205/java-8-liner-treemap
https://discuss.leetcode.com/topic/111244/simple-c-o-n-solution
https://discuss.leetcode.com/topic/111306/clean-c-o-logn-solution
LeetCode All in One 题目讲解汇总(持续更新中...)
[LeetCode] My Calendar I 我的日历之一的更多相关文章
- [LeetCode] My Calendar III 我的日历之三
		Implement a MyCalendarThree class to store your events. A new event can always be added. Your class ... 
- [LeetCode] My Calendar II 我的日历之二
		Implement a MyCalendarTwo class to store your events. A new event can be added if adding the event w ... 
- 与众不同 windows phone (26) - Contacts and Calendar(联系人和日历)
		原文:与众不同 windows phone (26) - Contacts and Calendar(联系人和日历) [索引页][源码下载] 与众不同 windows phone (26) - Con ... 
- [LeetCode] 731. My Calendar II 我的日历之二
		Implement a MyCalendarTwo class to store your events. A new event can be added if adding the event w ... 
- Java基础知识强化94:Calendar类之Calendar概述和获取日历字段的方法
		1. Calendar类概述: Calendar 类是一个抽象类,它为特定瞬间与一组诸如 YEAR.MONTH.DAY_OF_MONTH.HOUR 等 日历字段之间的转换提供了一些方法,并 ... 
- LeetCode My Calendar I
		原题链接在这里:https://leetcode.com/problems/my-calendar-i/description/ 题目: Implement a MyCalendar class to ... 
- java 的Calendar类的可视化日历示例
		import java.text.DateFormat; import java.text.ParseException; import java.text.SimpleDateFormat; imp ... 
- java基础1.5版后新特性 自动装箱拆箱 Date SimpleDateFormat Calendar.getInstance()获得一个日历对象 抽象不要生成对象 get set add System.arrayCopy()用于集合等的扩容
		8种基本数据类型的8种包装类 byte Byte short Short int Integer long Long float Float double Double char Character ... 
- 通过学习Date和Calendar时写的日历
		package com.etc.util; import java.util.Calendar; import java.util.Scanner; public class Calendar2 { ... 
随机推荐
- WHCTF-babyre
			WHCTF-babyre 首先执行file命令得到如下信息 ELF 64-bit LSB executable, x86-64 尝试用IDA64打开,定位到关键函数main发现无法F5,尝试了修复无果 ... 
- JavaScript(第二十七天)【错误处理与调试】
			JavaScript在错误处理调试上一直是它的软肋,如果脚本出错,给出的提示经常也让人摸不着头脑.ECMAScript第3版为了解决这个问题引入了try...catch和throw语句以及一些错误类型 ... 
- 敏捷冲刺每日报告四(Java-Team)
			第四天报告(10.28 周六) 团队:Java-Team 成员: 章辉宇(284) 吴政楠(286) 陈阳(PM:288) 韩华颂(142) 胡志权(143) github地址:https://gi ... 
- Week04-面向对象设计与继承
			1. 本周学习总结 1.1 写出你认为本周学习中比较重要的知识点关键词 is a关系 覆盖 Object 超级父类 继承 抽象类 多态 重载 static super private public p ... 
- 简单的C语言编译器--词法分析器
			1. 定义词法单元Tag 首先要将可能出现的词进行分类,可以有不同的分类方式.如多符一类:将所有逗号.分号.括号等都归为一类,或者一符一类,将一个符号归为一类.我这里采用的是一符一类的方式.C代码 ... 
- python控制流 If-else
			控制流 If-else 我们处理现实生活中的问题时会做出决定,就像决定买哪种相机或者怎样更好的打篮球.同样我们写计算机程序的时候也要做相同的事情.我们通过 if-else 语句来做决定,我们使 ... 
- find命令之(-atime,-ctime,-mtime)
			关于find命令,以拙见总结如下: >>>定义: find命令用来在指定目录下查找文件. 任何位于参数之前的字符串都将被视为欲查找的目录名.如果使用该命令时,不设置任何参数,则fin ... 
- Vue-cli+Vue.js2.0+Vuex2.0+vue-router+es6+webpack+node.js脚手架搭建和Vue开发实战
			Vue.js是一个构建数据驱动的web界面的渐进式框架.在写这边文章时Vue版本分为1.0++和2.0++,这个是基于Vue2.0的项目. Vue-cli是构建单页应用的脚手架,这个可是官方的. Vu ... 
- HDFS的7个设计特点
			1.Block的放置:默认不配置.一个Block会有三份备份,一份放在NameNode指定的DataNode,另一份放在与指定DataNode非同一Rack上的DataNode,最后一份放在与指定Da ... 
- System.Reflection名称空间下的程序集类Assembly应用.
			利用反射中的程序集类(Assembly--抽象类)动态加载类库(.dll)或者可执行程序(.exe). 优点:①.可以消除if条件的逻辑判断.②.减少内存资源.③.有利于程序扩展. 缺点... 使用静 ... 
