JZOJ 1322 硬币游戏
来源:互联网 发布:微信支付 java开发 编辑:程序博客网 时间:2024/06/05 20:24
题目大意:
FJ的奶牛喜欢玩硬币游戏,所以FJ发明了一个新的硬币游戏。一开始有N(5<=N<=2,000)个硬币堆成一叠,从上往下数第i个硬币有一个整数值C_i(1<=C_i<=100,000)。
两个玩家轮流从上倒下取硬币,玩家1先取,可以从上面取1个或2个硬币,下一轮的玩家可以取的硬币数量最少为1个,最多为上一个玩家取的数量的2倍,硬币全部取完比赛结束。
已知玩家2绝顶聪明,会采用最优策略,现在请你帮助玩家1,使得玩家1取得的硬币值的和最大。
题解:
要使得玩家一取得的硬币值最大,也可以说成使玩家一取得硬币值减去玩家二取得的硬币值最大,这样看起来就有些博弈函数的味道了,在后面的dp中会更加方便。
设
这样是O(
我们可以发现
显然这是O(
Code:
#include<cstdio>#define fo(i, x, y) for(int i = x; i <= y; i ++)#define fd(i, x, y) for(int i = x; i >= y; i --)#define min(a, b) ((a) < (b) ? (a) : (b))#define max(a, b) ((a) > (b) ? (a) : (b))using namespace std;const int Maxn = 2005;int n, c[Maxn], s[Maxn], f[Maxn][Maxn];int main() { scanf("%d", &n); fo(i, 1, n) scanf("%d", &c[i]); if(n == 1) { printf("%d\n", c[1]); return 0; } fo(i, 1, n) s[i] = s[i - 1] + c[i]; fd(i, n - 1, 1) { int max = -1e9; fo(j, 1, i) { fo(k, min(i + 2 * j - 2, n) + 1, min(i + 2 * j, n)) max = max(max, s[k] - s[i] - f[k][k - i]); f[i][j] = max; } } printf("%d", (s[n] + max(c[1] - f[1][1], c[1] + c[2] - f[2][2])) / 2);}
阅读全文
1 0
- JZOJ 1322 硬币游戏
- JZOJ 1322. 硬币游戏
- 硬币游戏
- 硬币游戏
- 硬币游戏
- 抛硬币游戏模拟
- 1349 翻硬币游戏
- 猜硬币游戏设计
- 博弈-翻硬币游戏
- 博弈-翻硬币游戏
- 博弈-翻硬币游戏
- 博弈-翻硬币游戏
- 翻硬币游戏
- 2015京东校招-硬币游戏
- bzoj1411: [ZJOI2009]硬币游戏
- bzoj1411 硬币游戏 分治
- 51nod1381 硬币游戏
- [bzoj4600][SDOI2016]硬币游戏
- python爬虫实战
- MySql触发器使用讲解
- kubernetes 概念&部署
- 更新xocde描述文件
- 懂二进制 小米2016年校招 Java实现
- JZOJ 1322 硬币游戏
- 【NOIP2015】斗地主题解
- [Java算法分析与设计]顺序循环队列的实现
- ecjtu-summer training #2 A
- 数据抽象to
- 希尔排序算法
- 远程访问服务器tensorboard的方法
- 【HNOI2012】bzoj2733 永无乡
- TIM简单的即时通信以及仿QQ聊天页面布局