codevs 3342 绿色通道
《思远高考绿色通道》(Green Passage, GP)是唐山一中常用的练习册之一,其题量之大深受lsz等许多oiers的痛恨,其中又以数学绿色通道为最。2007年某月某日,soon-if (数学课代表),又一次宣布收这本作业,而lsz还一点也没有写……
高二数学《绿色通道》总共有n道题目要写(其实是抄),编号1..n,抄每道题所花时间不一样,抄第i题要花a[i]分钟。由于lsz还要准备NOIP,显然不能成天写绿色通道。lsz决定只用不超过t分钟时间抄这个,因此必然有空着的题。每道题要么不写,要么抄完,不能写一半。一段连续的空题称为一个空题段,它的长度就是所包含的题目数。这样应付自然会引起马老师的愤怒。马老师发怒的程度(简称发怒度)等于最长的空题段长度。
现在,lsz想知道他在这t分钟内写哪些题,才能够尽量降低马老师的发怒度。由于lsz很聪明,你只要告诉他发怒度的数值就可以了,不需输出方案。(快乐融化:那么lsz怎么不自己写程序?lsz:我还在抄别的科目的作业……)
第一行为两个整数n,t,代表共有n道题目,t分钟时间。
以下一行,为n个整数,依次为a[1], a[2],... a[n],意义如上所述。
仅一行,一个整数w,为最低的发怒度。
17 11
6 4 5 2 5 3 4 5 2 3 4 5 2 3 6 3 5
3
60%数据 n<=2000
100%数据 0<n<=50000,0<a[i]<=3000,0<t<=100000000
最小化最大值——二分
很明显的一个DP
每次二分最长空题段,判断mid
方法一:朴素的DP
令f[i]表示抄第i道题所花费的最小时间
状态转移方程:f[i]=min(f[j])+time[i] max(0,i-mid-1)<=j<=i-1
初始化:f数组极大值,f[0]=0
时间复杂度:n²
#include<cstdio>
#include<cstring>
#include<algorithm>
#define N 50001
using namespace std;
int n,t,l,r,mid,ans,f[N],a[N];
inline bool check(int k)
{
memset(f,,sizeof(f));
f[]=;
for(int i=;i<=n;i++)
for(int j=max(i-k-,);j<i;j++)
f[i]=min(f[i],f[j]+a[i]);
int tmp=0x7fffffff;
for(int i=n-k;i<=n;i++) tmp=min(tmp,f[i]);
if(tmp<=t) return true;
return false;
}
int main()
{
scanf("%d%d",&n,&t);
for(int i=;i<=n;i++) scanf("%d",&a[i]);
l=,r=n;
while(l<=r)
{
mid=l+r>>;
if(check(mid)) {ans=mid;r=mid-;}
else l=mid+;
}
printf("%d",ans);
}
结果:
方法二:线段树优化DP
用线段树维护区间最小值,就可以直接查询[i-mid-1,i-1]内的最小值
时间复杂度:log n (二分)*n(枚举每道题)*(log n+log n)(区间查询+单点修改)= 2*n*log²n
结果:
#include<cstdio>
#include<algorithm>
#define N 50001
#define INF 100000010
using namespace std;
int n,t,ql,qr,mid,ans,f[N],a[N];
struct node{int l,r,key;}tr[N*];
inline int read()//读入优化
{
int x=;char c=getchar();
while(c<''||c>'') c=getchar();
while(c>=''&&c<='') {x=x*+c-'';c=getchar();}
return x;
}
inline void begin(int k,int l,int r)//初始化
{
tr[k].key=INF;
if(l==r) return;
int mid=l+r>>;
begin(k<<,l,mid);
begin((k<<)+,mid+,r);
}
inline int query(int k,int opl,int opr)//区间查询
{
if(tr[k].l>=opl&&tr[k].r<=opr) return tr[k].key;
int mid=tr[k].l+tr[k].r>>;
{
int ll=INF;if(opl<=mid) ll=query(k<<,opl,opr);
int rr=INF;if(opr>mid) rr=query((k<<)+,opl,opr);
return min(ll,rr);
}
}
inline void change(int k,int x,int y)//单点修改
{
if(tr[k].l==tr[k].r) {tr[k].key=min(tr[k].key,y);return;}
int mid=tr[k].l+tr[k].r>>;
if(x<=mid) change(k<<,x,y);
else change((k<<)+,x,y);
tr[k].key=min(tr[k<<].key,tr[(k<<)+].key);
}
inline bool check(int k)
{
begin(,,n);
change(,,);
for(int i=;i<=n;i++)
{
int p=query(,max(,i-k-),i-);
change(,i,p+a[i]);
}
int tmp=0x7fffffff;
if(query(,n-k,n)<=t) return true;
return false;
}
inline void build(int k,int l,int r)//建树
{
tr[k].l=l;tr[k].r=r;
if(l==r) return;
int mid=l+r>>;
build(k<<,l,mid);
build((k<<)+,mid+,r);
}
int main()
{
scanf("%d%d",&n,&t);
for(int i=;i<=n;i++) scanf("%d",&a[i]);
build(,,n);
ql=,qr=n;
while(ql<=qr)
{
mid=ql+qr>>;
if(check(mid)) {ans=mid;qr=mid-;}
else ql=mid+;
}
printf("%d",ans);
}
方法三:单调队列优化DP
假设有一段长度为mid的滑动窗口,每次这个窗口可以框住长为mid的一段数
我们需要的是长为mid的一段数里最小的那一个,所以可以维护一个单调上升的队列,这样每次直接取出队首就好了。
为什么要维护单调递增的队列?
因为如果这个数比前一个大,当窗口移过前一个数时,这个数可能成为最小的
如果这个数比前一个小,这个数的前一个一定不如这个优,所以删去大的
所以这就需要维护2个数组,1个存储队列中元素的在原序列中的位置(即第几道题),1个存储所需时间
如何维护这个队列的单调性?
1、数必须在窗口范围内,假设当前要计算第i个,那么可以更新它的范围是[i-mid-1,i-1],所以如果当前队首位置<i-mid-1,就让其出队
2、单调递增,如果第抄第i-1道题所需时间比当前的队尾时间少,就让队尾出队
3、在确定位置插入i-1的信息
4、用队首信息更新当前点信息
#include<cstdio>
#include<cstring>
#include<algorithm>
#define N 50001
using namespace std;
int n,t,head,tail,a[N],f[N],q[N],id[N],l,r,mid,ans;
//a存储抄每道题所需时间,f抄第i道题的最少时间,q单调队列数组,存储所需时间,id单调队列数组,存储题目编号
inline bool check(int k)
{
memset(q,,sizeof(q));
memset(f,,sizeof(f));
memset(id,,sizeof(id));
head=tail=;
//队列采取左闭右开式,有效范围为[head,tail-1]
for(int i=;i<=n;i++)
{
while(id[head]<i-k-&&head<tail) head++;//队首题目编号不在可以更新i的范围内,出队
while(f[i-]<q[tail-]&&head<tail) tail--;//不满足单调性,出队
q[tail]=f[i-];//入队
id[tail]=i-;
tail++;
f[i]=q[head]+a[i];
}
int tmp=0x7fffffff;
for(int i=n-k;i<=n;i++) tmp=min(tmp,f[i]);
if(tmp<=t) return true;
return false;
}
int main()
{
scanf("%d%d",&n,&t);
for(int i=;i<=n;i++) scanf("%d",&a[i]);
l=,r=n;
while(l<=r)
{
mid=l+r>>;
if(check(mid)) {ans=mid;r=mid-;}
else l=mid+;
}
printf("%d",ans);
}
结果:
codevs 3342 绿色通道的更多相关文章
- codevs 3342绿色通道
3342 绿色通道 时间限制: 1 s 空间限制: 256000 KB 题目等级 : 黄金 Gold
- AC日记——绿色通道 codevs 3342
3342 绿色通道 时间限制: 1 s 空间限制: 256000 KB 题目等级 : 黄金 Gold 题解 查看运行结果 题目描述 Description <思远高考绿色通道&g ...
- 绿色通道(codevs 3342)
题目描述 Description <思远高考绿色通道>(Green Passage, GP)是唐山一中常用的练习册之一,其题量之大深受lsz等许多oiers的痛恨,其中又以数学绿色通道为最 ...
- codevs3342绿色通道(单调队列优化dp)
3342 绿色通道 时间限制: 1 s 空间限制: 256000 KB 题目等级 : 黄金 Gold 题目描述 Description <思远高考绿色通道>(Green Pass ...
- codevs 3289 花匠
题目:codevs 3289 花匠 链接:http://codevs.cn/problem/3289/ 这道题有点像最长上升序列,但这里不是上升,是最长"波浪"子序列.用动态规划可 ...
- codevs 1082 线段树练习 3(区间维护)
codevs 1082 线段树练习 3 时间限制: 3 s 空间限制: 128000 KB 题目等级 : 大师 Master 题目描述 Description 给你N个数,有两种操作: 1:给区 ...
- codevs 1285 二叉查找树STL基本用法
C++STL库的set就是一个二叉查找树,并且支持结构体. 在写结构体式的二叉查找树时,需要在结构体里面定义操作符 < ,因为需要比较. set经常会用到迭代器,这里说明一下迭代器:可以类似的把 ...
- codevs 1576 最长上升子序列的线段树优化
题目:codevs 1576 最长严格上升子序列 链接:http://codevs.cn/problem/1576/ 优化的地方是 1到i-1 中最大的 f[j]值,并且A[j]<A[i] .根 ...
- codevs 1080 线段树点修改
先来介绍一下线段树. 线段树是一个把线段,或者说一个区间储存在二叉树中.如图所示的就是一棵线段树,它维护一个区间的和. 蓝色数字的是线段树的节点在数组中的位置,它表示的区间已经在图上标出,它的值就是这 ...
随机推荐
- 【Luogu3804】【模板】后缀自动机(后缀自动机)
[Luogu3804][模板]后缀自动机(后缀自动机) 题面 洛谷 题解 一个串的出现次数等于\(right/endpos\)集合的大小 而这个集合的大小等于所有\(parent\)树上儿子的大小 这 ...
- 【BZOJ2946】公共串(后缀数组)
[BZOJ2946]公共串(后缀数组) 题面 权限题... 只有CJOJ题面啦 Description 给出几个由小写字母构成的单词,求它们最长的公共子串的长度. 任务: 读入单词,计算最长公共子串的 ...
- [Luogu3066][USACO12DEC]逃跑的BarnRunning Away From…
题面 题目描述 给出以1号点为根的一棵有根树,问每个点的子树中与它距离小于等于l的点有多少个. 输入格式: Line 1: 2 integers, N and L (1 <= N <= 2 ...
- Poj2749:Building roads
题意 有 N 个牛栏,现在通过一条通道(s1,s2)要么连到s1,要么连到s2,把他们连起来,他们之间有一些约束关系,一些牛栏不能连在同一个点,一些牛栏必须连在同一个点,现在问有没有可能把他们都连好, ...
- c++面试遇到问题
1. C 和 C++ 区别 2. const 有什么用途 主要有三点: 1:定义只读变量,即常量 2:修饰函数的参数和函数的返回值 3: 修饰函数的定义体,这里的函数为类的成员函数, ...
- 笔记:基于DCNN的图像语义分割综述
写在前面:一篇魏云超博士的综述论文,完整题目为<基于DCNN的图像语义分割综述>,在这里选择性摘抄和理解,以加深自己印象,同时达到对近年来图像语义分割历史学习和了解的目的,博古才能通今!感 ...
- 接收JSON类型转成对象
写个小例子吧: public String getJsonTest(String jsonString){} 参数是json 参数长这样 ===> { 'puser' : {'id' : ' ...
- php 创建和修改文件内容
file_put_contents写入文件 我们先来学习第一种写入文件的方式: int file_put_contents ( string $文件路径, string $写入数据]) 功能:向指定的 ...
- jquery中的ajax方法参数
引用来自:http://www.cnblogs.com/tylerdonet/p/3520862.html jquery中的ajax方法参数总是记不住,这里记录一下. 1.url: 要求为String ...
- 删除项目中的.pyc文件
在编写python项目的时候,往往会生成很多的.pyc文件,但是在我们提交代码的时候这些.pyc文件又是不必要的,所以需要对项目里面的.pyc文件进行清理.再此,经过我的查阅资料和尝试,分享出以下方法 ...