【BZOJ 2436】 2436: [Noi2011]Noi嘉年华 (区间DP)
2436: [Noi2011]Noi嘉年华
Description
NOI2011 在吉林大学开始啦!为了迎接来自全国各地最优秀的信息学选手,
吉林大学决定举办两场盛大的 NOI 嘉年华活动,分在两个不同的地点举办。每
个嘉年华可能包含很多个活动,而每个活动只能在一个嘉年华中举办。
现在嘉年华活动的组织者小安一共收到了 n个活动的举办申请,其中第 i 个
活动的起始时间为 Si,活动的持续时间为Ti。这些活动都可以安排到任意一个嘉
年华的会场,也可以不安排。
小安通过广泛的调查发现,如果某个时刻,两个嘉年华会场同时有活动在进
行(不包括活动的开始瞬间和结束瞬间),那么有的选手就会纠结于到底去哪个
会场,从而变得不开心。所以,为了避免这样不开心的事情发生,小安要求不能
有两个活动在两个会场同时进行(同一会场内的活动可以任意进行)。
另外,可以想象,如果某一个嘉年华会场的活动太少,那么这个嘉年华的吸
引力就会不足,容易导致场面冷清。所以小安希望通过合理的安排,使得活动相
对较少的嘉年华的活动数量最大。
此外,有一些活动非常有意义,小安希望能举办,他希望知道,如果第i 个
活动必须举办(可以安排在两场嘉年华中的任何一个),活动相对较少的嘉年华
的活动数量的最大值。Input
输入的第一行包含一个整数 n,表示申请的活动个数。
接下来 n 行描述所有活动,其中第 i 行包含两个整数 Si、Ti,表示第 i 个活
动从时刻Si开始,持续 Ti的时间。Output
输出的第一行包含一个整数,表示在没有任何限制的情况下,活动较少的嘉
年华的活动数的最大值。
接下来 n 行每行一个整数,其中第 i 行的整数表示在必须选择第 i 个活动的
前提下,活动较少的嘉年华的活动数的最大值。Sample Input
5
8 2
1 5
5 3
3 2
5 3Sample Output
2
2
1
2
2
2HINT
在没有任何限制的情况下,最优安排可以在一个嘉年华安排活动 1, 4,而在
另一个嘉年华安排活动 3, 5,活动2不安排。
1≤n≤200 0≤Si≤10^9
1≤Ti≤ 10^9
Source
【分析】
怎么说,又不会做。。
这个两个东东的最小值最大不会搞,额,当然这里是不能二分的嘛。。
也不会在DP中记录,然后题解的方法好像。。称呼其为“定一议二”?就是DP中有一维说的是其中一个人拿了x个区间的情况下,另一个人最多拿多少区间。【很对吧!
这样子应该是会做第一问了的,n^3DP就可以过。
但是后面,一个东西必须做的,就会感觉要n^4吧。
但是有类似单调性的东西,这个后面再详细讲,网上他们都没有仔细说,我是自己推了一下的。
先看这个大神的详细题解,这个写得真的很清晰很好懂啊!
来自:http://blog.csdn.net/qpswwww/article/details/45251877
然后说说那个什么递增单凸的。
首先,显然pre[][x]和suf[][y]都是递减的。
对于x确定,y在变,f[x][y]=min(x+y,pre[i][x]+num[i][j]+suf[j][y]),显然x+y随y递增而增,pre[i][x]+num[i][j]+suf[j][y]随y递增而减。
就是这样的,下面标红的函数就是真正的函数,显然是上凸的了。
所以程序里面y按顺序,找到一个now<当前最优值 就可以break了。
然后说明一个就是随着x的增加,取最优值的y单调递减。这个画个图也可以看出来了。
所以就是这样做了,y这里均摊的话,就是O(n^3)
【调了一晚上好内伤,我好蠢啊。。
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define Maxn 210
#define INF 0xfffffff int n,p;
int mymax(int x,int y) {return x>y?x:y;}
int mymin(int x,int y) {return x<y?x:y;} int s[Maxn],t[Maxn],num[*Maxn][*Maxn];
int pre[*Maxn][*Maxn],suf[*Maxn][*Maxn],f[*Maxn][*Maxn]; struct node {int x,y;}a[*Maxn];
bool cmp(node x,node y) {return x.x<y.x;} void init()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%d%d",&s[i],&t[i]);
t[i]+=s[i];
a[i*-].x=s[i];a[i*-].y=i;
a[i*].x=t[i];a[i*].y=-i;
}
sort(a+,a++n*,cmp);
p=;
for(int i=;i<=n*;i++)
{
if(i==||a[i].x!=a[i-].x) p++;
if(a[i].y>) s[a[i].y]=p;
else t[-a[i].y]=p;
}
memset(num,,sizeof(num));
memset(pre,,sizeof(pre));
memset(suf,,sizeof(suf));
for(int i=;i<=p;i++)
for(int j=i;j<=p;j++)
for(int k=;k<=n;k++) if(s[k]>=i&&t[k]<=j) num[i][j]++; for(int i=;i<=p;i++)
for(int j=;j<=n;j++)
for(int k=;k<=i;k++)
{
if(j>num[][i]) {pre[i][j]=-INF;continue;}
pre[i][j]=mymax(pre[i][j],pre[k][j]+num[k][i]);
if(j>=num[k][i]) pre[i][j]=mymax(pre[i][j],pre[k][j-num[k][i]]);
}//printf("\n");
for(int i=p;i>=;i--)
for(int j=;j<=n;j++)
for(int k=i;k<=p;k++)
{
if(j>num[i][p]) {suf[i][j]=-INF;continue;}
suf[i][j]=mymax(suf[i][j],suf[k][j]+num[i][k]);
if(j>=num[i][k]) suf[i][j]=mymax(suf[i][j],suf[k][j-num[i][k]]);
}
} int main()
{
init(); int ans=;
for(int i=;i<=p;i++)
for(int j=i;j<=p;j++)
{
int y=num[j][p];
for(int x=;x<=num[][i];x++)
{
int id;
for(;y>=;y--)
{
int nw=mymin(x+y,pre[i][x]+num[i][j]+suf[j][y]);
if(f[i][j]<=nw)
{
f[i][j]=nw;
id=y;
}
else break;
}
y=id;
}
ans=mymax(ans,f[i][j]);
} for(int i=;i<=p;i++)
for(int j=p;j>=i;j--) f[i][j]=mymax(f[i][j],f[i][j+]);
for(int i=;i<=p;i++)
for(int j=i;j<=p;j++) f[i][j]=mymax(f[i][j],f[i-][j]);
printf("%d\n",ans);
for(int i=;i<=n;i++)
printf("%d\n",f[s[i]][t[i]]); return ;
}
2017-03-22 21:47:30
【BZOJ 2436】 2436: [Noi2011]Noi嘉年华 (区间DP)的更多相关文章
- BZOJ2436 [Noi2011]Noi嘉年华 【dp】
题目链接 BZOJ2436 题解 看这\(O(n^3)\)的数据范围,可以想到区间\(dp\) 发现同一个会场的活动可以重叠,所以暴力求出\(num[l][r]\)表示离散化后\([l,r]\)的完整 ...
- 2436: [Noi2011]Noi嘉年华 - BZOJ
Description NOI2011 在吉林大学开始啦!为了迎接来自全国各地最优秀的信息学选手,吉林大学决定举办两场盛大的 NOI 嘉年华活动,分在两个不同的地点举办.每个嘉年华可能包含很多个活动, ...
- bzoj 2436: [Noi2011]Noi嘉年华
Description NOI2011 在吉林大学开始啦!为了迎接来自全国各地最优秀的信息学选手,吉林大学决定举办两场盛大的 NOI 嘉年华活动,分在两个不同的地点举办.每个嘉年华可能包含很多个活动, ...
- luogu P1973 [NOI2011]NOI 嘉年华 dp
LINK:NOI 嘉年华 一道质量非常高的dp题目. 考虑如何求出第一问 容易想到dp. 按照左端点排序/右端点排序状态还是很难描述. 但是我们知道在时间上肯定是一次选一段 所以就可以直接利用时间点来 ...
- BZOJ.4897.[Thu Summer Camp2016]成绩单(区间DP)
BZOJ 显然是个区间DP.令\(f[l][r]\)表示全部消掉区间\([l,r]\)的最小花费. 因为是可以通过删掉若干子串来删子序列的,所以并不好直接转移.而花费只与最大最小值有关,所以再令\(g ...
- bzoj 1055 [HAOI2008]玩具取名(区间DP)
1055: [HAOI2008]玩具取名 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1258 Solved: 729[Submit][Statu ...
- BZOJ 1260: [CQOI2007]涂色paint( 区间dp )
区间dp.. dp( l , r ) 表示让 [ l , r ] 这个区间都变成目标颜色的最少涂色次数. 考虑转移 : l == r 则 dp( l , r ) = 1 ( 显然 ) s[ l ] = ...
- BZOJ 1996: [Hnoi2010]chorus 合唱队(区间dp)
题目: https://www.lydsy.com/JudgeOnline/problem.php?id=1996 题解: 这题刚拿到手的时候一脸懵逼qwq,经过思考与分析(看题解),发现是一道区间d ...
- BZOJ.3928.[CERC2014]Outer space invaders(区间DP)
BZOJ3928 双倍经验BZOJ4048 Codeforces GYM100543 L \(Description\) \(Solution\) 考虑出现时间在\([l,r]\)内的敌人,设最远的敌 ...
随机推荐
- net 加密-解密
#region DES加密 解密 //key:32位 public string DESEncrypt(string strSource, byte[] key) { System.Security. ...
- webpack4 未设置mode会自动压缩
最近想用LayaBox做个小游戏,然而Laya本身不自带构建工具.然后觉得写模块化的东西还是用webpack好使,用es6的语法也比较清晰. 于是就装了webpack,只用babel-loader来编 ...
- 线程句柄和线程ID的区别
●CreateThread() API 用于创建线程. API 返回同时线程句柄,并通过参数得到线程标识符 (ID). 线程句柄有完全访问权创建线程对象. 运行线程时线程 ID 唯一标识线程在系统级别 ...
- 【C++自我精讲】基础系列六 PIMPL模式
[C++自我精讲]基础系列六 PIMPL模式 0 前言 很实用的一种基础模式. 1 PIMPL解释 PIMPL(Private Implementation 或 Pointer to Implemen ...
- php菜刀分析学习
这里以eval为例 我们知道, php中的eval能把字符串当代码执行: eval('phpcode'); 注意, 这里的代码要有分号结尾, 我们测试: 我们创建一个最简单的SHELL: <?p ...
- 14 - 函数参数检测-inspect模块
目录 1 python类型注解 2 函数定义的弊端 3 函数文档 4 函数注解 4.1 annotation属性 5 inspect模块 5.1 常用方法 5.2 signature类 5.3 par ...
- C/C++——C语言库函数大全
本文转载自:https://blog.csdn.net/yanfan0916/article/details/6450442###; 1. 分类函数: ctype.h int isalpha(int ...
- mysql 操作时间戳
1.将long显示成时间 SELECT FROM_UNIXTIME(1249488000, '%Y%m%d' ) 2.日期格式化成时间戳 SELECT UNIX_TIMESTAMP('2016-05- ...
- 欧拉回路&欧拉通路判断
欧拉回路:图G,若存在一条路,经过G中每条边有且仅有一次,称这条路为欧拉路,如果存在一条回路经过G每条边有且仅有一次, 称这条回路为欧拉回路.具有欧拉回路的图成为欧拉图. 判断欧拉通路是否存在的方法 ...
- python基础(9)--递归、二叉算法、多维数组、正则表达式
1.递归 在函数内部,可以调其他函数,如果一个函数在内部调用它本身,这个函数就是递归函数.递归算法对解决一大类问题是十分有效的,它往往使算法的描述简洁而且易于裂解 递归算法解决问题的特点: 1)递归是 ...