NOIP2012借教室
在大学期间,经常需要租借教室。大到院系举办活动,小到学习小组自习讨论,都需要
向学校申请借教室。教室的大小功能不同,借教室人的身份不同,借教室的手续也不一样。
面对海量租借教室的信息,我们自然希望编程解决这个问题。
我们需要处理接下来n天的借教室信息,其中第i天学校有ri个教室可供租借。共有m份
订单,每份订单用三个正整数描述,分别为dj, sj, tj,表示某租借者需要从第sj天到第tj天租
借教室(包括第sj天和第tj天),每天需要租借dj个教室。
我们假定,租借者对教室的大小、地点没有要求。即对于每份订单,我们只需要每天提
供dj个教室,而它们具体是哪些教室,每天是否是相同的教室则不用考虑。
借教室的原则是先到先得,也就是说我们要按照订单的先后顺序依次为每份订单分配教
室。如果在分配的过程中遇到一份订单无法完全满足,则需要停止教室的分配,通知当前申
请人修改订单。这里的无法满足指从第sj天到第tj天中有至少一天剩余的教室数量不足dj个。
现在我们需要知道,是否会有订单无法完全满足。如果有,需要通知哪一个申请人修改
订单。
第一行包含两个正整数n, m,表示天数和订单的数量。
提高组 day2
第二行包含n个正整数,其中第i个数为ri,表示第i天可用于租借的教室数量。
接下来有m行,每行包含三个正整数dj, sj, tj,表示租借的数量,租借开始、结束分别在
第几天。
每行相邻的两个数之间均用一个空格隔开。天数与订单均用从1开始的整数编号。
如果所有订单均可满足,则输出只有一行,包含一个整数 0。否则(订单无法完全满足)
输出两行,第一行输出一个负整数-1,第二行输出需要修改订单的申请人编号。
4 3
2 5 4 3
2 1 3
3 2 4
4 2 4
-1
2
【输入输出样例说明】
classroom.out
-1
2
第 1 份订单满足后,4 天剩余的教室数分别为 0,3,2,3。第 2 份订单要求第 2 天到
第 4 天每天提供 3 个教室,而第 3 天剩余的教室数为 2,因此无法满足。分配停止,通知第
2 个申请人修改订单。
【数据范围】
对于 10%的数据,有1 ≤ n, m ≤ 10;
对于 30%的数据,有1 ≤ n, m ≤ 1000;
对于 70%的数据,有1 ≤ n, m ≤ 105;
对于 100%的数据,有1 ≤ n, m ≤ 10^6, 0 ≤ ri, dj≤ 10^9, 1 ≤ sj≤ tj≤ n。、
最开始看到这道题的时候是在刷线段树专题的时候,当时老师说借教室这道题可以用线段树水过,下面贴上写的非常丑的线段树做法
#include<stdio.h>
#include<algorithm>
using namespace std;
int h[];
struct node
{
int l,r,min,sign;
}tree[];
void build(int now,int l,int r)
{
int mid=(l+r)/;
tree[now].l=l;
tree[now].r=r;
if(l==r)
{
tree[now].min=h[l];
return;
}
build(now*,l,mid);
build(now*+,mid+,r);
tree[now].min=min(tree[now*].min,tree[now*+].min);
}
void putdown(int now)
{
tree[now*].sign+=tree[now].sign;
tree[now*+].sign+=tree[now].sign;
tree[now*].min-=tree[now].sign;
tree[now*+].min-=tree[now].sign;
tree[now].sign=;
}
bool gai(int now,int l,int r,int sum)
{
if(tree[now].l>r||tree[now].r<l) return true;
if(tree[now].l>=l&&tree[now].r<=r)
{
tree[now].min-=sum;
tree[now].sign+=sum;
if(tree[now].min<) return false;
return true;
}
if(tree[now].l!=tree[now].r&&tree[now].sign!=) putdown(now);
if((gai(now*,l,r,sum)==)||(gai(now*+,l,r,sum)==))
{
tree[now].min=min(tree[now*].min,tree[now*+].min);
return false;
}
tree[now].min=min(tree[now*].min,tree[now*+].min);
return true;
}
int main()
{
int i,j,dj,sj,tj,n,m;
scanf("%d %d",&n,&m);
for(i=;i<=n;i++)
scanf("%d",&h[i]);
build(,,n);
for(j=;j<=m;j++)
{
scanf("%d %d %d",&dj,&sj,&tj);
if(gai(,sj,tj,dj)==)
{
printf("-1\n%d",j);
return ;
}
}
printf("");
return ;
}
现在看看当初写的线段树真是不能再暴力了。
下面说正解,这道题是求最小的天数,使所有人都可以借完教室。天数多,可能不符合;但天数少,也未必不行。
所以用二分,check的时候用前缀和表示某一天借出的教室数
比如前缀和最开始是0 0 0 0 0
第3天到第5天借出2个教室
数列就变成0 0 2 0 -2
最后计算数列中每天的借出教室的值是否超过准备的教室
代码写的比较丑,凑合着看看吧
#include<stdio.h>
#include<algorithm>
#include<string.h>
using namespace std;
const int mn=;
int num[mn],R[mn],d[mn],s[mn],t[mn],m,n,ans;
template<class T>void read(T &x)
{
x=;char ch=getchar();
while(ch<''||ch>'') ch=getchar();
while(ch>=''&&ch<=''){x=(x<<)+(x<<)+(ch^);ch=getchar();}
return;
}
bool check(int a)
{
int sum=;
memset(num,,sizeof(num));
for(int i=;i<=a;i++)
{
num[s[i]]+=d[i];
num[t[i]+]-=d[i];
}
for(int i=;i<=n;i++)
{
sum+=num[i];
if(sum>R[i]) return ;
}
return ;
}
int main()
{
read(n),read(m);
for(int i=;i<=n;++i)
read(R[i]);
for(int i=;i<=m;++i)
read(d[i]),read(s[i]),read(t[i]);
int l=,r=m;
while(l<=r)
{
int mid=(l+r)>>;
if(check(mid)) l=mid+;
else {ans=mid;r=mid-;}
}
if(ans==) printf("");
else printf("-1\n%d",ans);
return ;
}
NOIP2012借教室的更多相关文章
- NOIP2012借教室[线段树|离线 差分 二分答案]
题目描述 在大学期间,经常需要租借教室.大到院系举办活动,小到学习小组自习讨论,都需要 向学校申请借教室.教室的大小功能不同,借教室人的身份不同,借教室的手续也不一样. 面对海量租借教室的信息,我们自 ...
- NC16564 [NOIP2012]借教室
NC16564 [NOIP2012]借教室 题目 题目描述 在大学期间,经常需要租借教室.大到院系举办活动,小到学习小组自习讨论,都需要向学校申请借教室.教室的大小功能不同,借教室人的身份不同,借 ...
- NOIP2012 借教室
描述 在大学期间,经常需要租借教室.大到院系举办活动,小到学习小组自习讨论,都需要向学校申请借教室.教室的大小功能不同,借教室人的身份不同,借教室的手续也不一样.面对海量租借教室的信息,我们自然希望编 ...
- NOIP2012 借教室 Splay初探
终于把区间操作的Splay搞明白了…… Splay的大致框架是这样的: [代码中的Zig-Zig和Zig-Zag操作其实是可以优化的,实际只需要3次passDown和3次update] templat ...
- 【洛谷P1083】[NOIP2012]借教室
借教室 [题目描述] 在n天中每天有一个可以借出的教室数,有m个订单,每个订单从第l天到第r天要借用x个教室.问能否满足所有的订单,若不能,输出第一个不能满足的订单编号. 思路: 1.1 ≤ n,m ...
- [NOIP2012]借教室 题解
题目大意: 有一个n个数的数列,m个操作,第i个操作使[li,ri]区间建di,问第几个操作使数列中出现负数. 思路: 暴力显然过不了,那么就可以优化了,不难想到线段树,显然需要良好的姿势,那么就差分 ...
- luogu1083 [NOIp2012]借教室 (二分答案+差分)
先二分一个答案x,然后通过差分来看有没有不满足的 #include<bits/stdc++.h> #define pa pair<int,int> #define lowb(x ...
- NOIP2012 借教室 题解 洛谷P1083
一看就是暴力 好吧,其实是线段树或差分+二分,这里用的是差分+二分的做法. 二分部分的代码,套个二分板子就行 ,right=m; while(left<right)//二分 { ; ; else ...
- 洛谷 1083 (NOIp2012) 借教室——标记永久化线段树 / 差分+二分
题目:https://www.luogu.org/problemnew/show/P1083 听说线段树不标记永久化会T一个点. 注意mn记录的是本层以下.带上标记的min! #include< ...
随机推荐
- r里面如何实现两列数据合并为一列
library(dplyr) unite(mtcars, "vs_am", vs, am) Merging Data Adding Columns To merge two dat ...
- 设计模式——proxy代理模式
目录 概述 定义 角色 为什么会有代理模式? 应用场景 示例 静态代理 例子 动态代理 JDK中生成代理对象的API 代码示例: 代码示例2 Cglib代理 代码示例 AOP(AspectOrient ...
- [转]谈NAND Flash的底层结构和解析
这里我想以一个纯玩家的角度来谈谈关于NAND Flash的底层结构和解析,可能会有错误的地方,如果有这方面专家强烈欢迎指正. NAND Flash作为一种比较实用的固态硬盘存储介质,有自己的一些物理特 ...
- http 请求 post get 长度限制
一.问题起因在某项目释放后Bug统计的附件<释放后问题>里有: 问题 原因 分析 备注 CSV处理时,如果处理的主题数过多,发生URL参数上限的错误: 可变长度的参数通过UR ...
- .net读取Excel转datatable、.net读取的Excel存在合并单元格并且转成datatable
项目中经常会遇到Excel导入数据,Excel的模板会可能是存在合并单元格的,模板如下图所示 读取时需要填充合并单元格的值,转成datatable单元格值时,填充合并单元格的值,如下图所示: 合并单元 ...
- C#使用xpath简单爬取网站的内容
public static void Get() { // string xpathtrI = "//*[@id='classify-list']/dl/dd/a/cite/span/i&q ...
- 关于vb编程之字符串连接/拼接的方法与技巧
在VB中,笔者知道的字符串的拼接方法主要有两种拼接符号,一种为"&"符,另一种则为"+"符 一.其中&连接运算符用于强制将两个表达式作为字符串连 ...
- mssql for xml path使用
准备工作: CREATE TABLE [dbo].[Students]( [id] [int] IDENTITY(1,1) NOT NULL, [names] [varchar](50) NULL, ...
- python文档自动翻译
关键方法 提取文档内容 读取TXT文档 txt文档的读取很简单,直接用python自带的open()方法就好,代码如下所示: # 读取TXT文档 def read_txt(path): '''实现TX ...
- 第三天 Linux简单命令
2018-5-22 15:21:59 使用 atom 可以在windows环境下同步代码与linux (汉化配置好就可以啦) 2018-4-13 18:09:31 该看32节啦 1.man +陌生命 ...