首先我们得贪心一下,让吃饭时间较长的人排在队首

去抄一段贪心的证明吧

现在研究在一个队伍里的人有什么性质

可以发现道题里也有一个不变量,就是对于队伍里的前\(i\)个人,不管他们排队的顺序如何,\(a[i]\)的前缀和\(s[i]\)是不会变的,第\(i\)个人会在\(s[i]+b[i]\)时刻吃完,这前\(i\)个人的吃饭时间是\(max(s[j]+b[j])\)

现在可以回到贪心的问题上了,假设对于两个相邻的人\(i\)和\(i+1\),如果\(a[i]==a[i+1]\),那么显然让\(b[i]>b[i+1]\)好

由上面的性质可以看到\(s[i]\)是守恒的,那么让第\(i\)个人的\(b[i]\)是前\(i\)个人里最小的当然就是坠吼的啦

之后就可以dp了

感觉自己dp的姿势不太对

最开始yy出来的状态是\(dp[i][j]\)表示前\(i\)个人里\(j\)个人去了一个队列时消耗时间最长的人的最小耗时

但是这个状态显然不对,因为我们不知道这\(j\)个人的总的等待时间,好想没有什么办法进行转移

于是缺一个状态就加上一维吧,那就用\(dp[i][j][k]\)表示前\(i\)个人里\(j\)个人去了一个队列时等待时间为\(k\)消耗时间最长的人的最小耗时

发现这样的话空间好像要炸的样子,尽管开滚动好像都不太行

但是又发现第二维好像没有什么存的必要,因为转移的时候根本就跟这个队列里有多少人没有关系

于是最后的状态就是\(dp[i][j]\)表示前\(i\)个人里一个队列总等待时间为\(k\)消耗时间最长的人的最小耗时

我们考虑一下如何转移,因为这两个队列本质是相同的,所以一个人有两种选择,一个就是进入在状态里表示的那个队列,在一个就是进入另一个队列

进入状态里表示的这个队列的话,这个队列的总时长会增加,等待时间最长的人可能也会有变化

于是就有

\[dp[i][j+a[i]]=max(dp[i-1][j],j+a[i]+b[i])
\]

进入另外一个队列,这个队列的等待时长没有变化,但是我们需要知道另外一个队列所有人的总等待时间,由于我闷知道这个队列的总等待时间是\(j\),那么我们用一个前缀和就能求出来另外一个队列的总等待时间

\[dp[i][j]=max(dp[i-1][j],pre[i-1]-j+a[i]+b[i])
\]

代码

  1. #include<iostream>
  2. #include<cstring>
  3. #include<algorithm>
  4. #include<cstdio>
  5. #define re register
  6. #define maxn 205
  7. #define INF 336860180
  8. #define max(a,b) ((a)>(b)?(a):(b))
  9. #define min(a,b) ((a)<(b)?(a):(b))
  10. struct Node
  11. {
  12. int a,b;
  13. }t[maxn];
  14. int n;
  15. int dp[201][200*200+1];
  16. int pre[maxn];
  17. int o;
  18. inline int read()
  19. {
  20. char c=getchar();
  21. int x=0;
  22. while(c<'0'||c>'9') c=getchar();
  23. while(c>='0'&&c<='9')
  24. x=(x<<3)+(x<<1)+c-48,c=getchar();
  25. return x;
  26. }
  27. inline int cmp(Node AA,Node BB)
  28. {
  29. return AA.b>BB.b;
  30. }
  31. int main()
  32. {
  33. n=read();
  34. for(re int i=1;i<=n;i++)
  35. t[i].a=read(),t[i].b=read();
  36. std::sort(t+1,t+n+1,cmp);
  37. for(re int i=1;i<=n;i++)
  38. pre[i]=pre[i-1]+t[i].a;
  39. memset(dp,20,sizeof(dp));
  40. dp[1][t[1].a]=t[1].a+t[1].b;
  41. for(re int i=2;i<=n;i++)
  42. {
  43. for(re int j=0;j<=200*200;j++)
  44. {
  45. if(dp[i-1][j]==INF) continue;
  46. dp[i][j+t[i].a]=min(max(dp[i-1][j],j+t[i].a+t[i].b),dp[i][j+t[i].a]);
  47. dp[i][j]=min(dp[i][j],max(dp[i-1][j],pre[i-1]-j+t[i].a+t[i].b));
  48. }
  49. }
  50. int ans=INF;
  51. for(re int i=1;i<=200*200;i++)
  52. ans=min(ans,dp[n][i]);
  53. std::cout<<ans;
  54. return 0;
  55. }

【[ZJOI2005]午餐】的更多相关文章

  1. [洛谷P2577] [ZJOI2005]午餐

    洛谷题目链接:[ZJOI2005]午餐 题目描述 上午的训练结束了,THU ACM小组集体去吃午餐,他们一行N人来到了著名的十食堂.这里有两个打饭的窗口,每个窗口同一时刻只能给一个人打饭.由于每个人的 ...

  2. [ZJOI2005]午餐 (DP)

    [ZJOI2005]午餐 题目描述 上午的训练结束了,THU ACM小组集体去吃午餐,他们一行N人来到了著名的十食堂.这里有两个打饭的窗口,每个窗口同一时刻只能给一个人打饭.由于每个人的口味(以及胃口 ...

  3. 洛谷P2577 [ZJOI2005]午餐 打饭时间作为容量DP

    P2577 [ZJOI2005]午餐 )逼着自己做DP 题意: 有n个人打饭,每个人都有打饭时间和吃饭时间.有两个打饭窗口,问如何安排可以使得总用时最少. 思路: 1)可以发现吃饭时间最长的要先打饭. ...

  4. Luogu P2577 [ZJOI2005]午餐(dp)

    P2577 [ZJOI2005]午餐 题面 题目描述 上午的训练结束了, \(THU \ ACM\) 小组集体去吃午餐,他们一行 \(N\) 人来到了著名的十食堂.这里有两个打饭的窗口,每个窗口同一时 ...

  5. 2018.09.22 ZJOI2005午餐(贪心+01背包)

    描述 上午的训练结束了,THU ACM小组集体去吃午餐,他们一行N人来到了著名的十食堂.这里有两个打饭的窗口,每个窗口同一时刻只能给一个人打饭.由于每个人的口味(以及胃口)不同,所以他们要吃的菜各有不 ...

  6. P2577 [ZJOI2005]午餐 状压DP

    题目描述 上午的训练结束了,THU ACM小组集体去吃午餐,他们一行N人来到了著名的十食堂.这里有两个打饭的窗口,每个窗口同一时刻只能给一个人打饭.由于每个人的口味(以及胃口)不同,所以他们要吃的菜各 ...

  7. P2577 [ZJOI2005]午餐

    题目描述 上午的训练结束了,THU ACM小组集体去吃午餐,他们一行N人来到了著名的十食堂.这里有两个打饭的窗口,每个窗口同一时刻只能给一个人打饭.由于每个人的口味(以及胃口)不同,所以他们要吃的菜各 ...

  8. 【洛谷P2577】[ZJOI2005]午餐

    午餐 题目链接 DP题都辣么毒瘤的么.. 首先,看一下题解 我们就有了思路: 贪心:显然,让吃饭慢的先打饭,sort一遍(证明?不存在的.. DP:f[i][j][k]表示前i个人,窗口1的打饭时间为 ...

  9. [ZJOI2005]午餐 (贪心,动态规划)

    题目描述 上午的训练结束了,THU ACM小组集体去吃午餐,他们一行N人来到了著名的十食堂.这里有两个打饭的窗口,每个窗口同一时刻只能给一个人打饭.由于每个人的口味(以及胃口)不同,所以他们要吃的菜各 ...

  10. P2577 [ZJOI2005]午餐[DP]

    题目描述 上午的训练结束了,THU ACM小组集体去吃午餐,他们一行N人来到了著名的十食堂.这里有两个打饭的窗口,每个窗口同一时刻只能给一个人打饭.由于每个人的口味(以及胃口)不同,所以他们要吃的菜各 ...

随机推荐

  1. response提交原理(转)

    摘自:http://blog.csdn.net/quechao123/article/details/6256653         http://jorton468.blog.163.com/blo ...

  2. 出现多个sessid

    这种情况我也出现了,很郁闷.这是每次请求页面,都生成一个存PHPSESSID的cookie,cookie路径的问题,可以发现你这几个PHPSESSID的Path都不一样.我修改Thinkphp的配置文 ...

  3. spring的事务传播行为

    1.PROPAGATION_REQUIRED:如果当前没有事务,就创建一个新事务,如果当前存在事务,就加入该事务,该设置是最常用的设置. 比如说,ServiceB.methodB的事务级别定义为PRO ...

  4. JavaWeb之JSP原理

    1.为什么需要JSP? 在很多动态网页中,绝大部分内容都是固定不变的,只有局部内容需要动态产生和改变.如果使用Servlet程序来输出只有局部内容需要改动的网页,其中所有的静态内容也需要程序员用jav ...

  5. Nginx 504错误总结

    Nginx 504错误(Gateway time-out  网关超时)的含义是所请求的网关没有请求到,简单来说就是没有请求到可以执行的PHP-CGI. 一般看来, 这种情况可能是由于nginx默认的f ...

  6. Redis数据类型及常用命名总结

    Redis数据类型: Redis支持五种数据类型:string(字符串),hash(哈希),list(列表),set(集合)及zset(sorted set:有序集合).  1.String(字符串) ...

  7. git基本命令集合

    以下内容不适合初学者 括号中表示需要自己填写 v1.0 git add git commit -m git commit -a -m git commit -amend git clone git l ...

  8. CSS3自定义loading效果

    效果: 使用CSS3完成loading的制作 css样式: <style type="text/css"> .mask { position: fixed; left: ...

  9. 响应式(2)——bootstrap的响应式

    <meta name="viewport" content="width=device-width,user-scalable=no"/> < ...

  10. Java BeanUtils 组件 使用

    1. BeanUtils组件 1.1 简介 程序中对javabean的操作很频繁, 所以apache提供了一套开源的api,方便对javabean的操作!即BeanUtils组件. BeanUtils ...