Description

A 公司正在举办一个智力双人游戏比赛----取石子游戏,游戏的获胜者将会获得 A 公司提供的丰厚奖金,因此吸引了来自全国各地的许多聪明的选手前来参加比赛。

与经典的取石子游戏相比,A公司举办的这次比赛的取石子游戏规则复杂了很多: 
  总共有N堆石子依次排成一行,第i堆石子有 ai个石子。 
  开始若干堆石子已被 A公司故意拿走。

  然后两个玩家轮流来取石子,每次每个玩家可以取走一堆中的所有石子,但有一个限制条件:一个玩家若要取走一堆石子,则与这堆石子相邻的某堆石子已被取走(之前被某个玩家取走或开始被A公司故意拿走)。注意:第 1堆石子只与第 2堆石子相邻,第N堆石子只与第N-1堆石子相邻,其余的第 i堆石子与第i-1堆和第 i+1 堆石子相邻。

  所有石子都被取走时,游戏结束。谁最后取得的总石子数最多,谁就获得了这场游戏的胜利。

作为这次比赛的参赛者之一,绝顶聪明的你,想知道对于任何一场比赛,如果先手者和后手者都使用最优的策略,最后先手者和后手者分别能够取得的总石子数分别是多少。

Input

第一行是一个正整数N,表示有多少堆石子。输入文件第二行是用空格隔开的N个非负整数a1, a2, …, aN,其中ai表示第i堆石子有多少个石子,

ai  = 0表示第i堆石子开始被A公司故意拿走。输入的数据保证0≤ai≤100,000,000,并且至少有一个i使得ai = 0。30%的数据满足2≤N≤100,100%的数据满足2≤N≤1,000,000。

Output

仅包含一行,为两个整数,分别表示都使用最优策略时,最后先手者和后手者各自能够取得的总石子数,并且两个整数间用一个空格隔开。

Sample Input

8
1 2 0 3 7 4 0 9

Sample Output

17 9
样例解释:两个玩家都使用最优策略时取走石子的顺序依次为9, 2, 1, 4, 7, 3,因此先手者取得9 + 1 + 7 = 17个石子,后手者取得2 + 4 + 3 = 9个石子。

正解:博弈论+链表。

这道题的难点在于分析博弈者的心理。

双方的目标就是使自己-别人的石子数差最大化。

可以看出,石子堆分为两个栈和若干个双端队列。

如果所有元素在每一个容器中都是递减的,那么先手和后手显然可以每次都取最大元素来保证自己利益的最大化。

但是现在不是这样,不过我们可以把这个数列化简一下。

如果序列最左端是$A,B...$或最右端是$...B,A$,且$B\leq A$。

那么双方在有其它方案时都不会愿意先取走$B$,故这种情况可以留到博弈的最后。

由于石子数是确定的,可以直接推出最后谁取到了$A$,算出相应差值。

由于可以留到游戏的最后,此时删除这两堆并不影响两人之前的决策。

如果有一段 $...A,B,C...$且满足$A\leq B,C\leq B$。

那么我们直接把 ABC 替换成一个 A+C-B 即可。

我们可以这样想:选$A,B,C$的时候是因为没有更好的决策而被迫选的。事实上当全场没有大于$A+C-B$的石子堆可以直接取时,才会考虑取$A,C$中的一个。那么不管第一次取$A,B,C$中的元素是从哪边,后手一定也没有别的更好的选择,既然先手选$A,C$都已是被迫了,所以后手选$B$一定不会是差的。留下来的一个也一定是当前不差的选择。故先手一定取走$A+C$,后手取走$B$。从对分数差的贡献来看,我们可以直接把$A,B,C$代替成$A+C-B$。

我们用链表实现化简,就能把石子变成递减的情况,然后直接排序取石子就行了。

 #include <bits/stdc++.h>
#define il inline
#define RG register
#define ll long long
#define N (1000005) using namespace std; int b[N],l[N],r[N],n,h,t,ok,cnt,tmp;
ll a[N],c[N],sum,dif; il int gi(){
RG int x=,q=; RG char ch=getchar();
while ((ch<'' || ch>'') && ch!='-') ch=getchar();
if (ch=='-') q=-,ch=getchar();
while (ch>='' && ch<='') x=x*+ch-,ch=getchar();
return q*x;
} il void del(RG int x){
RG int L=l[x],R=r[x];
if (L) r[L]=R,l[x]=;
if (R<=n) l[R]=L,r[x]=n+;
return;
} int main(){
#ifndef ONLINE_JUDGE
freopen("stone.in","r",stdin);
freopen("stone.out","w",stdout);
#endif
n=gi(),h=,t=n,ok=;
for (RG int i=;i<=n;++i)
a[i]=gi(),sum+=a[i],b[i]=!a[i],cnt^=!b[i],l[i]=i-,r[i]=i+;
while (ok && h<t){
ok=;
while (h<t && !b[h] && !b[r[h]] && a[h]>=a[r[h]]){
cnt?dif+=a[h]-a[r[h]]:dif+=a[r[h]]-a[h];
tmp=r[r[h]],del(r[h]),del(h),h=tmp,ok=;
}
while (h<t && !b[t] && !b[l[t]] && a[t]>=a[l[t]]){
cnt?dif+=a[t]-a[l[t]]:dif+=a[l[t]]-a[t];
tmp=l[l[t]],del(l[t]),del(t),t=tmp,ok=;
}
for (RG int i=h;i<=t;i=r[i]){
if (b[i] || !l[i] || r[i]==n+ || b[l[i]] || b[r[i]]) continue;
if (a[i]<a[l[i]] || a[i]<a[r[i]]) continue;
if (h==l[i]) h=i; if (t==r[i]) t=i;
a[i]=a[l[i]]+a[r[i]]-a[i],del(l[i]),del(r[i]),ok=;
}
}
cnt=; for (RG int i=h;i<=t;i=r[i]) if (!b[i]) c[++cnt]=a[i];
sort(c+,c+cnt+),reverse(c+,c+cnt+);
for (RG int i=;i<=cnt;++i) i&?dif+=c[i]:dif-=c[i];
printf("%lld %lld\n",(sum+dif)>>,(sum-dif)>>); return ;
}

最新文章

  1. springMvc的搭建
  2. Mac 下使用homebrew 安装node后全局安装找不到问题
  3. 深入理解javascript函数系列第四篇——ES6函数扩展
  4. Codeforces 715A. Plus and Square Root[数学构造]
  5. 【CodeForces 602B】G - 一般水的题2-Approximating a Constant Range
  6. log4j---------学习总结(一)
  7. Linux下软件包的多种安装方式
  8. sqlserver insert into select
  9. asp.net 给按钮 增加事件
  10. bzoj 3594: [Scoi2014]方伯伯的玉米田
  11. ubuntu18.04 安装mysql 5.7.22
  12. Team
  13. EntityFramework(1)基础概念与Database First
  14. Tensorflow object detection API 搭建物体识别模型(一)
  15. Python logging模块简介
  16. PyQT5速成教程-4 Qt Designer实战[上]
  17. odoo开发笔记 -- odoo10 视图界面根据字段状态,动态隐藏创建&amp;编辑按钮
  18. Android的几种弹出框
  19. SQL Server默认1433端口修改方法
  20. vbox虚拟机扩容(CentOS 7.2)

热门文章

  1. MyBatis Mapper XML 文件 的学习详解
  2. https Java SSLException protocol_version的问题解决方法
  3. (原创).Net将EF运用于Oralce一 准备工作
  4. [android] 界面切换的简单动画
  5. 一 NIO的概念
  6. Python中元组和列表
  7. java设计模式之抽象工厂模式学习
  8. Effective C++ .14 智能指针的拷贝与deleter函数
  9. float、display和流
  10. 类型同时存在于A.dll和B.dll中的解决办法