博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
[洛谷P1430]序列取数
阅读量:5320 次
发布时间:2019-06-14

本文共 1316 字,大约阅读时间需要 4 分钟。

题目大意:给定一个序列$s$,每个人每轮可以从两端(任选一端)取任意个数的整数,不能不取。在两个人都足够聪明的情况下,求先手的最大得分。

题解:设$f_{i,j}$表示剩下$[i,j]$,先手的最大得分。令$sum_{i,j}=\sum\limits_{k=i}^j s_k$

$$\therefore f_{i,j}=sum_{i,j}-\min\{\min\limits_{k=j-1}^i f_{i,k},\min\limits_{k=i+1}^j f_{k,j},0\}$$

这是$O(n^3)$的,会$TLE$

$$令l_{i,j}=\min\limits_{k=i}^j f_{i,k}, r_{i,j}=\min\limits_{k=i}^j f_{k,j}$$

$$f_{i,j}=sum_{i,j}-\min\{l_{i,j-1},r_{i+1,j},0\}$$

时间复杂度:$O(n^2)$

卡点:

 

C++ Code:

#include 
#include
#define maxn 1010using namespace std;int Tim, n;int s[maxn], sum[maxn], f[maxn][maxn];int l[maxn][maxn], r[maxn][maxn];inline int min(int a, int b) {return a < b ? a : b;}inline int max(int a, int b) {return a > b ? a : b;}int main() { scanf("%d", &Tim); while (Tim --> 0) { memset(l, 0x3f, sizeof l); memset(r, 0x3f, sizeof r); scanf("%d", &n); for (int i = 1; i <= n; i++) { scanf("%d", &s[i]); sum[i] = sum[i - 1] + s[i]; } for(int i = 1; i <= n; i++) { for(int j = i; j; j--) { int &t = f[j][i], tmp; t = sum[i] - sum[j - 1]; tmp = min(l[j][i - 1], r[j + 1][i]); t = max(t, t - tmp); if(j == i) l[j][i] = r[j][i] = t; else { l[j][i] = min(l[j][i - 1], t); r[j][i] = min(r[j + 1][i], t); } } } printf("%d\n", f[1][n]); } return 0;}

  

转载于:https://www.cnblogs.com/Memory-of-winter/p/9492734.html

你可能感兴趣的文章
sqlserver 各种判断是否存在(表名、函数、存储过程等)
查看>>
给C#学习者的建议 - CLR Via C# 读后感
查看>>
Recover Binary Search Tree
查看>>
Java 实践:生产者与消费者
查看>>
[转]IOCP--Socket IO模型终结篇
查看>>
js 获取视频的第一帧
查看>>
各种正则验证
查看>>
观察者模式(Observer)
查看>>
python中numpy.r_和numpy.c_
查看>>
egret3D与2D混合开发,画布尺寸不一致的问题
查看>>
freebsd 实现 tab 命令 补全 命令 提示
查看>>
struts1和struts2的区别
查看>>
函数之匿名函数
查看>>
shell习题第16题:查用户
查看>>
实验4 [bx]和loop的使用
查看>>
Redis常用命令
查看>>
2018.11.06 bzoj1040: [ZJOI2008]骑士(树形dp)
查看>>
2019.02.15 bzoj5210: 最大连通子块和(链分治+ddp)
查看>>
redis cluster 集群资料
查看>>
微软职位内部推荐-Sr. SE - Office incubation
查看>>