题目

测试地址:POJ2228USACO 2005 January Gold

题目来源:USACO 2005 January Gold / 模拟赛 T2(本题题面)。

题目描述

正在 rainbow 的城堡游玩的 Freda 恰好看见了在地毯上跳舞卖萌的水叮当……于是……

Freda:“呜咕>_< 我也要卖萌T_T!”

rainbow 给了 Freda \(N\) 秒的自由活动时间,不过由于刚刚游览城堡有些累了,Freda 只想花 \(B\) 秒的时间来卖萌,剩下的时间她要在 rainbow 的城堡里睡个好觉好好休息一下。

rainbow 给这 \(N\) 秒每秒定义了一个值 \(U_i\),如果第 \(i\) 秒钟 Freda 在卖萌,那么她可以获得 \(U_i\) 点卖萌指数lala~

Freda 开始卖萌后可以随时停止,休息一会儿之后再开始。不过每次 Freda 开始卖萌时,都需要 \(1\) 秒来准备= =,这一秒是不能获得卖萌指数的。当然,Freda 卖萌和准备的总时间 不能超过 \(B\)。

更特殊的是,这 \(N\) 秒钟时间是 环形 的。也就是 Freda 可以从任意时间开始她的自由活动并持续 \(N\) 秒。

为了使自己表现得比水叮当更萌,现在 Freda 想知道,她最多能获得多少卖萌指数呢?

输入格式

第一行包含两个整数 \(N\) 和 \(B\)。

第 \(2\)~\(N+1\) 行每行一个整数,其中第 \(i+1\) 行的整数表示 \(U_i\)。

输出格式

输出一个整数,表示 Freda 可以获得的最大卖萌指数。

数据范围

各个测试点时间限制 \(1\mathrm{s}\),空间限制 \(1\mathrm{GiB}\)。

  • 对于 \(60\%\) 的数据,\(N\leq 100\)
  • 对于 \(100\%\) 的数据,\(0\leq B\leq N\leq 3600\),\(0\leq U_i\leq 200000\)。

分析

这道题可以看出来是 \(\mathtt{DP}\),只不过环形这个条件比较难搞。

线性情况

状态定义和递推

先考虑线性的情况,定义 \(f_{i,j}\) 为前 \(i\) 分钟已经卖萌 \(j\) 分钟的卖萌指数最大值。

则可以得到:(\(val(i,j)\)代表从 \(i\) 时刻到 \(j\) 时刻的卖萌指数)

\[\huge{f_{i,j} = \max\{f_{i-1,j}, \max_{0< k\leq j}\{f_{i-k,j-k}+val(i-k+1,i)\}\}}
\]

在这个基础上,我们发现这个算法的复杂度是 \(\Theta(n^3)\) 的,只能通过部分分。如何优化?

优化

首先,\(val(i,j)\) 是可以通过前缀和实现 \(\Theta(1)\) 的(设前缀和数组为\(\{pref_i\}\))。接下来考虑 \(\max_{0< k\leq j}\{f_{i-k,j-k}\}\)

考虑数列 \(l_{k,u} = \max_{k\leq t\leq u}\{f_{t,t-k}-pref_{t+1}\}\),则 \(\max_{0< k\leq j}\{f_{i-k,j-k}\}\) 就是 \(l_{i-j,i-1}+pref_i\)。

每次计算出 \(f_{i,j}\) 后,可以 \(\Theta(1)\) 更新 \(l_{i-j,i}\),这样就可以将复杂度降至 \(\Theta(n^2)\)。

同时 \(l_{k,u}\) 可以简化成 \(l_k\),节省空间。(虽然这道题有的是空间)

上代码:

  1. for (int i = 1; i <= n; i++)
  2. for (int j = 0; j <= i && j <= lim; j++)
  3. {
  4. dp[i][j] = ab_max(dp[i-1][j], ln[i-j] + pref[i]);
  5. if (i != n)
  6. ln[i-j] = ab_max(ln[i-j], dp[i-1][j] - pref[i+1]);
  7. }

环形情况

既然线性情况 Get,那么环形又如何处理呢?

对于最优情况,首先考虑这几个结论:

  • 卖萌结束后的下一分钟,卖萌必然不开始。

为什么?如果下一分钟开始了,这一分钟的卖萌指数就无法算入。但是如果将这一分钟连上上一分钟的卖萌,这一分钟的卖萌指数就能算入。

考虑到 \(U_i \geq 0\),所以算上一定 不差于 不算上。

  • 线性与环形的唯一差别就是第 \(1\) 分钟有没有卖萌 (不是准备)

为什么?因为如果第 \(1\) 分钟没有卖萌或在做准备,则与线性的某种情况等价。

唯一要考虑的环形情况就是有跨越的卖萌时间段。

所以,我们再进行一遍 \(\mathtt{DP}\)。只不过这一次 \(\mathtt{DP}\) 强制要求第 \(1\) 分钟正在卖萌,而且从某一时刻到第 \(n\) 分钟一直在卖萌(有准备)。

为了思路清晰,这次使用 \(revdp_{i,j}\) 进行记录。

上代码:

  1. memset(ln, 0xcf, sizeof(ln));
  2. revdp[0][0] = revdp[1][0] = 0;
  3. ln[0] = -pref[1]; // 注意初始化
  4. revdp[0][0] = 0;
  5. for (int i = 1; i <= lim; i++)
  6. revdp[i][i] = pref[i];
  7. for (int i = 1; i < n; i++)
  8. for (int j = 0; j < i && j <= lim; j++)
  9. {
  10. revdp[i][j] = ab_max(revdp[i-1][j], ln[i-j] + pref[i]);
  11. ln[i-j] = ab_max(ln[i-j], revdp[i-1][j] - pref[i+1]);
  12. }
  13. for (int i = 0; i <= lim; i++)
  14. revdp[n][i] = ln[n-i] + pref[n];

最后只需要在 \(dp_{n,b}\) 和 \(revdp_{n,b}\) 中选择一个最小值就行了。

Code

  1. #include <cstdio>
  2. #include <cstring>
  3. using namespace std;
  4. const int max_n = 3600;
  5. int pref[max_n+1], dp[max_n+1][max_n+1], ln[max_n+1], revdp[max_n+1][max_n+1] = {};
  6. inline int ab_max(int x, int y) { return (x > y)? x:y; }
  7. int main()
  8. {
  9. memset(dp, 0xcf, sizeof(dp));
  10. memset(revdp, 0xcf, sizeof(revdp));
  11. memset(ln, 0xcf, sizeof(ln));
  12. dp[0][0] = dp[1][0] = 0;
  13. int n, lim, tmp;
  14. scanf("%d%d", &n, &lim);
  15. if (lim <= 1)
  16. {
  17. puts("0");
  18. return 0;
  19. }
  20. pref[0] = 0;
  21. for (int i = 0; i < n; i++)
  22. {
  23. scanf("%d", &tmp);
  24. pref[i+1] = pref[i] + tmp;
  25. }
  26. ln[0] = -pref[1];
  27. for (int i = 1; i <= n; i++)
  28. for (int j = 0; j <= i && j <= lim; j++)
  29. {
  30. dp[i][j] = ab_max(dp[i-1][j], ln[i-j] + pref[i]);
  31. if (i != n)
  32. ln[i-j] = ab_max(ln[i-j], dp[i-1][j] - pref[i+1]);
  33. }
  34. memset(ln, 0xcf, sizeof(ln));
  35. revdp[0][0] = revdp[1][0] = 0;
  36. ln[0] = -pref[1];
  37. revdp[0][0] = 0;
  38. for (int i = 1; i <= lim; i++)
  39. revdp[i][i] = pref[i];
  40. for (int i = 1; i < n; i++)
  41. for (int j = 0; j < i && j <= lim; j++)
  42. {
  43. revdp[i][j] = ab_max(revdp[i-1][j], ln[i-j] + pref[i]);
  44. ln[i-j] = ab_max(ln[i-j], revdp[i-1][j] - pref[i+1]);
  45. }
  46. for (int i = 0; i <= lim; i++)
  47. revdp[n][i] = ln[n-i] + pref[n];
  48. printf("%d\n", ab_max(dp[n][lim], revdp[n][lim]));
  49. return 0;
  50. }

备注

标算使用的定义是 \(f_{i,j,k}\) 为前 \(i\) 分钟共卖萌 \(j\) 分钟,第 \(i\) 分钟的状态为 \(k\) 时的最大卖萌指数。

则状态转移就是 \(f_{i,j,k}=\begin{cases}\max_{u=0,1}\{f_{i-1,j,u}\}&k=0\\\max\{f_{i-1,j-1,0},f_{i-1,j-1,1}+U_i\}\}&k=1\end{cases}\)

至于具体的细节,供读者思考。(明明是 5ab 太菜了,想不出来)

【题解】Acting Cute的更多相关文章

  1. Poetize6: Acting Cute

    3042: Acting Cute Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 59  Solved: 36[Submit][Status] Des ...

  2. BZOJ ac100题存档

    不知不觉AC100题了,放眼望去好像都是水题.在这里就做一个存档吧(特别感谢各位大神尤其是云神http://hi.baidu.com/greencloud和丽洁姐http://wjmzbmr.com/ ...

  3. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  4. 越狱Season 1-Episode 4: Cute Poison

    Season 1, Episode 4: Cute Poison [Previously on Prison Break] previously: 以前地 前情提要 -Burrows: I didn' ...

  5. Codeforces Round #447 (Div. 2) 题解 【ABCDE】

    BC都被hack的人生,痛苦. 下面是题解的表演时间: A. QAQ "QAQ" is a word to denote an expression of crying. Imag ...

  6. CodeChef November Challenge 2013 部分题解

    http://www.codechef.com/NOV13 还在比...我先放一部分题解吧... Uncle Johny 排序一遍 struct node{ int val; int pos; }a[ ...

  7. 2016女生专场 ABCDEF题解 其他待补...

    GHIJ待补... A.HUD5702:Solving Order Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/3276 ...

  8. Codeforces Round#630 div2 A~C题解

                                                                                                        ...

  9. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

随机推荐

  1. java学习-初级入门-面向对象②-面向对象概述-面向对象程序设计

    我们在  面向对象①中学习了,结构化程序设计. 今天我们一起学习面向对象程序设计. 学习面向对象程序设计就要了解,在面向对象中重要的知识点. 继承  .  多态   . 抽象  . 接口 我们会在接下 ...

  2. 【STM32H7教程】第53章 STM32H7的LTDC应用之汉字小字库和全字库制作

    完整教程下载地址:http://www.armbbs.cn/forum.php?mod=viewthread&tid=86980 第53章       STM32H7的LTDC应用之汉字小字库 ...

  3. 嵊州普及Day3T1

    题意:n座山,每天袭击k面,不能为同一座.问最少几天袭击所有山两面. 思路:不管如何,n,k<=10,做了就能过,考试时先想的暴力模拟,后来发现有规律,看看就好了. 见代码: #include& ...

  4. js获取cookie提取用户名asp.net+html

    JavaScript是运行在客户端的脚本,因此一般是不能够设置Session的,因为Session是运行在服务器端的. 而cookie是运行在客户端的,所以可以用JS来设置cookie. 假设有这样一 ...

  5. DevOps - 自动化工具

    章节 DevOps – 为什么 DevOps – 与传统方式区别 DevOps – 优势 DevOps – 不适用 DevOps – 生命周期 DevOps – 与敏捷方法区别 DevOps – 实施 ...

  6. greenplum 存储过程

    参考: https://docs.pivotal.io/search?q=function   官网 https://www.cnblogs.com/greenZ/p/8722081.html htt ...

  7. 1_01_MSSQL课程_基础入门2

    1.数据库的迁移方案 ->分离 附加 ->权限问题: ->启用Administrator账号 ->把数据库文件放到默认的数据库文件存放目录. ->数据库文件兼容级别,设置 ...

  8. 学习进度-10 python爬虫

    学习爬虫的第一个案例是小说爬虫. 小说爬虫首先是解析小说页面源代码,在页面源代码中可以看到小说每章节的内容链接 爬虫的代码: import requests import re url = 'http ...

  9. MongoDB 监控指标

    MongoDB uptime 启动时长 asserts.user 用户的断言数量 asserts.warning 警告的断言数量 connections.current 当前的连接数 大于 650co ...

  10. generator 和 yield

    yield 的使用 generator 生成器 yield 可以使生成器返回多次 我习惯于从表象推测,不喜欢官方文档,写的字都认识,结果变成句子之后,就一句都看不懂 所以先举一个例子来看一下这个东西怎 ...