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< ...
随机推荐
- opencv wlsfilter depth refinement demo
参考 https://docs.opencv.org/3.2.0/d3/d14/tutorial_ximgproc_disparity_filtering.html // This file is p ...
- [springBoot系列]--springBoot注解大全[转]
一.注解(annotations)列表 @SpringBootApplication:包含了@ComponentScan.@Configuration和@EnableAutoConfiguration ...
- 群晖搭建webssh
拷贝工程到系统根,然后需要赋予权限 sudo chmod 777 -R WebSSH2/ git clone https://github.com/zhaocundang/WebSSH2.git de ...
- dubbo RPC超时异常小结
dubbo消费者调用服务超时的原因可能有很多,今天排查问题花了两个小时,也查了很多资料,好像每一篇资料都是提出一个问题,所以简单总结几点: 1. 配置才是重中之重,仔细检查服务提供方的dubbo se ...
- 库存秒杀问题-redis解决方案- 接口限流
<?php/** * Created by PhpStorm. * redis 销量超卖秒杀解决方案 * redis 文档:http://doc.redisfans.com/ * ab -n 1 ...
- MySQL 全文检索 ngram插件
InnoDB全文索引:N-gram Parser[转] MySql5.7 建立全文索引 InnoDB默认的全文索引parser非常合适于Latin,因为Latin是通过空格来分词的.但对于像中文,日文 ...
- Nessus中文报告自动化脚本
前言 Nessus扫描完成,总要花挺多时间去整理报告,为此写了一个小脚本,用于自动化生成中文漏洞报告. 解析导出的html报告,自动翻译成中文,并提供修复建议,减少整理报告的时间,提升工作效率. gi ...
- python 算术运算
1. 算术运算符与优先级 # -*- coding:utf-8 -*- # 运算符含有+,-,*,/,**,//,% # ** 表示^ , 也就是次方 a = 2 ** 4 print '2 ** 4 ...
- Java中的堆内存设置对线程创建数的影响以及-Xss参数的记录
Java的线程对象是存储在堆上的,所以,能够创建多少个线程,受到堆空间的大小限制,同时也受到每个线程的大小的限制,假如线程对象内部有一个非常大的数组字段,那就非常影响能够创建的线程的大小 我们的例子: ...
- 禅道迁移(windows_to_linux)
需求分析 随着禅道数据的增加,原来通过虚拟机提供的mysql服务器相应速度跟不上需求.且原来禅道的前端与数据库分离安装在windows与linux中,现在提供实体服务器,需要将禅道环境迁移. 确认环境 ...