【SOJ 385】旅馆
【题目】
题目描述:
奶牛们最近的旅游计划,是到苏必利尔湖畔,享受那里的湖光山色,以及明媚的阳光。作为整个旅游的策划者和负责人,贝茜选择在湖边的一家著名的旅馆住宿。这个巨大的旅馆一共有 N (1 ≤ N ≤ 50,000) 间客房,它们在同一层楼中顺次一字排开,在任何一个房间里,只需要拉开窗帘,就能见到波光粼粼的湖面。
贝茜一行,以及其他慕名而来的旅游者,都是一批批地来到旅馆的服务台,希望能订到 D_i (1 ≤ D_i ≤ N) 间连续的房间。服务台的接待工作也很简单:如果存在 r 满足编号为 r..r+D_i-1 的房间均空着,他就将这一批顾客安排到这些房间入住;如果没有满足条件的 r,他会道歉说没有足够的空房间,请顾客们另找一家宾馆。如果有多个满足条件的 r,服务员会选择其中最小的一个。
旅馆中的退房服务也是批量进行的。每一个退房请求由 2 个数字 X_i、D_i 描述,表示编号为 X_i..X_i+D_i-1 (1 ≤ X_i ≤ N-D_i+1) 房间中的客人全部离开。退房前,请求退掉的房间中的一些,甚至是所有,可能本来就无人入住。
而你的工作,就是写一个程序,帮服务员为旅客安排房间。你的程序一共需要处理 M (1 ≤ M < 50,000) 个按输入次序到来的住店或退房的请求。第一个请求到来前,旅店中所有房间都是空闲的。
输入格式:
第 1 行: 2 个用空格隔开的整数:N、M 。
第 2.. M+1行: 第 i+1 描述了第 i 个请求,如果它是一个订房请求,则用 2 个数字 1、D_i 描述,数字间用空格隔开;如果它是一个退房请求,用 3 个以空格隔开的数字 2、X_i、D_i 描述.
输出格式:
对于每个订房请求,输出 1 个独占 1 行的数字:如果请求能被满足,输出满足条件的最小的 r;如果请求无法被满足,输出 0 。
样例数据:
输入
10 6 1 3 1 3 1 3 1 3 2 5 5 1 6
输出
1 4 7 0 5
【分析】
话说这道题数据是真的水,我用暴力加一点小优化都 A 了
不过数据强一点的话,O()的暴力肯定是会 T 掉的,我们要用一些更高效的算法
方法一:用线段树
下面我们用 0 表示这件房间还没有人住,1 表示这件房间已经有人住了
对于线段树的每个区间,我们要维护一下三个信息:
(1)L:表示该区间最左边开始最多有多少个连续的 0 。
(2)R:表示该区间最右边开始最多有多少个连续的 0 。
(3)maxn:表示该区间最长的连续的 0 的个数。
维护了这些信息,答案就很好求了
还有,由于是区间修改,我们要用到懒标记
时间复杂度:O()
方法二:用分块
分块要维护的东西和线段树一样
然后分块的话没什么好说的,很暴力的算法
时间复杂度:O()
【代码】
用线段树的代码:
#include<cstdio>
#include<cstring>
#include<algorithm>
#define N 50005
using namespace std;
int L[4*N],R[4*N],maxn[4*N];
bool change[4*N];
void build(int root,int l,int r)
{if(l==r){L[root]=1;R[root]=1;maxn[root]=1;return;}int mid=(l+r)>>1;build(root<<1,l,mid);build(root<<1|1,mid+1,r);L[root]=L[root<<1]+L[root<<1|1];R[root]=R[root<<1]+R[root<<1|1];maxn[root]=maxn[root<<1]+maxn[root<<1|1];
}
void pushdown(int root,int l,int r)
{int mid=(l+r)>>1;if(maxn[root]!=0){maxn[root<<1]=L[root<<1]=R[root<<1]=mid-l+1;maxn[root<<1|1]=L[root<<1|1]=R[root<<1|1]=r-mid;}else{maxn[root<<1]=L[root<<1]=R[root<<1]=0;maxn[root<<1|1]=L[root<<1|1]=R[root<<1|1]=0;}change[root]=false;change[root<<1]=change[root<<1|1]=true;
}
int find(int root,int l,int r,int x)
{if(l==r) return l;int mid=(l+r)>>1;if(change[root])pushdown(root,l,r);if(L[root]>=x) return l;if(maxn[root<<1]>=x) return find(root<<1,l,mid,x);if(R[root<<1]+L[root<<1|1]>=x) return mid-R[root<<1]+1;return find(root<<1|1,mid+1,r,x);
}
void update(int root,int l,int r,int mid)
{int temp=max(maxn[root<<1],maxn[root<<1|1]);maxn[root]=max(temp,R[root<<1]+L[root<<1|1]);if(L[root<<1]!=mid-l+1) L[root]=L[root<<1];else L[root]=L[root<<1]+L[root<<1|1];if(R[root<<1|1]!=r-mid) R[root]=R[root<<1|1];else R[root]=R[root<<1]+R[root<<1|1];
}
void modify(int root,int l,int r,int x,int y,int k)
{if(l>=x&&r<=y){if(!k) L[root]=R[root]=maxn[root]=0;else L[root]=R[root]=maxn[root]=r-l+1;change[root]=true;return;}int mid=(l+r)>>1;if(change[root]) pushdown(root,l,r);if(x<=mid) modify(root<<1,l,mid,x,y,k);if(y>mid) modify(root<<1|1,mid+1,r,x,y,k);update(root,l,r,mid);
}
int main()
{int n,m,i,s,x,y,ans;scanf("%d%d",&n,&m);build(1,1,n);for(i=1;i<=m;++i){scanf("%d",&s);if(s==1){scanf("%d",&x);if(maxn[1]<x)printf("0\n");else{ans=find(1,1,n,x);printf("%d\n",ans);modify(1,1,n,ans,ans+x-1,0);}}else{scanf("%d%d",&x,&y);modify(1,1,n,x,x+y-1,1);}}return 0;
}
分块的代码就不发了吧(主要是懒得写)
【SOJ 385】旅馆相关推荐
- SOJ 4543 4542
http://acm.scu.edu.cn/soj/problem.action?id=4542 递归用数组保存中间值 #include <cstdio> #include <cma ...
- Soj题目分类 python代码)
正值期末复习,刷点soj放松下 但想看看能不能在找点关于数据结构的题目来做一下. 在网上看到有不少人上传过那些关于部分SOJ题目的描述,但是说实话有些乱 不过我看到有个网页中包含的一个类似文档的东西, ...
- 考研计算机385分什么水平,机械专硕初试385分 给学弟学妹分享一下准备的过程(数学二)...
楼主21年考研,机械专硕,初试成绩385分,给大家分享一下考研初试的经验 楼主本科是南京某独立院校,报考的是江西某普通本科,数学是从19年十月份开始准备的,前前后后,大概准备了一年多,跟的是某虫系统班 ...
- 【机器学习基础】太棒了!这里有385篇自然语言处理和机器学习领域的综述总结...
文章来源于python遇见NLP,作者自然语言er 综述论文对于初学者来说,可以方便其在研究初期尽快掌握该方向的发展趋势.目前我在GitHub上发现一个开源项目,上面总结了385篇自然语言处理和机器学 ...
- LeetCode 第 34 场双周赛(385/2842,前13.5%)
文章目录 1. 比赛结果 2. 题目 1. LeetCode 5491. 矩阵对角线元素的和 easy 2. LeetCode 5492. 分割字符串的方案数 medium 3. LeetCode 5 ...
- 希尔伯特旅馆实验(文末送书)
数学的世界浩瀚广博,其中"无穷"的世界更是引人入胜.小孩子从学数数开始便会渐渐明白,数字的世界是无穷的,找不到尽头.拥有无穷多房间的酒店是什么样子?饼干罐里又藏着怎样的" ...
- 1593: [Usaco2008 Feb]Hotel 旅馆
1593: [Usaco2008 Feb]Hotel 旅馆 Time Limit: 10 Sec Memory Limit: 64 MB Submit: 489 Solved: 272 [Subm ...
- bzoj1593 [Usaco2008 Feb]Hotel 旅馆(线段树)
1593: [Usaco2008 Feb]Hotel 旅馆 Time Limit: 10 Sec Memory Limit: 64 MB Submit: 758 Solved: 419 [Subm ...
- HNU 实验五松雅的旅馆
文章目录 题目 思路 AC代码 题目 思路 暴力模拟吧..注意在最左边的旅馆的左边以及最右边旅馆的右边也可以建立旅馆,即可以越过区间建立.这是数组类题目常有的问题. AC代码 #include < ...
- (转载)适合大学生旅游时住的各地旅馆
2019独角兽企业重金招聘Python工程师标准>>> 适合大学生旅游时住的各地旅馆(很需要,怕丢,收藏)... 2008-12-22 01:01:38 来自: 梨-冰天雪地 标题: ...
最新文章
- 《C++ Primer Plus 6th》读书笔记 - 第8章 函数探幽
- linux不保存退出命令_面试提问说出16个linux命令,能凑齐不!高频22个Linux命令在这里...
- 计算机中的信息表示 ppt模板,计算机中信息的表示.ppt
- CF A. DZY Loves Hash
- WebService入门讲解
- 重新开始我的园子生活了
- oracle备份恢复
- 【BUG记录】> Android dependency ‘androidx.vectordrawable:vectordrawable‘ has different
- AI开发者十问:10分钟了解AI开发的基本过程
- 笨办法学 Python · 续 练习 3:质量
- bread是可数还是不可数_为什么英语里的面包bread是不可数名词?听老师给你讲语法,一听就明白了...
- 携程正式挂牌港交所 开盘涨近5%
- Scikit-learn快速入门教程和实例(一)(二)
- 《EMU8086安装报告》
- Kotlin教程,从入门到精通
- html本地视频在线播放,html5本地播放器
- java学习总结(16.07.16)Random类和BigDecimal类
- php数据结构 链表,php数据结构-单链表
- RK3288RK3399 GMAC以太网调试
- 阅读总结(计算机专业大学生的注意事项)
热门文章
- 设计模式04—工厂模式
- 操盘手“本来生活”,这样把“褚橙”卖成“励志橙”
- Python的raw string原始字串转化为string一般字符串,还有结尾插入‘\‘的方法,还有把string转化为raw string
- dtmf拨号原理matlab,matlab综合实验dtmf拨号器设计.doc
- O2O新猜想:如果商家这样做,还需要团购平台吗
- 95后,我们一起看过的剧
- macOS:隐私设置与数据库相关的问题
- 【矩阵计算GPU加速】numpy 矩阵计算利用GPU加速,cupy包
- Node art-template 和 prase-pody配置
- 用matlab表白,用函数表达对她的爱