【AGC002 E】Candy Piles
本来实在写不动这题 sol 了,但一想这是个经典的模型转化问题,于是就写了(.jpg)
题意
有一个序列 \(a_i\)。
两人轮流操作,每次操作为二选一:
1. 把最大的 \(a_i\) 减成 \(0\)
2. 把所有非 \(0\) 的 \(a_i\) 减去 \(1\)
若一个人操作后,所有 \(a_i\) 都是 \(0\),这个人就输了。
两人都采用轮流策略,问谁能赢。
\(n\le 10^5\)
\(a_i\le 10^9\)
题解
智商模型转化:把所有 \(a_i\) 从大到小排序,画一个柱状图,第 \(i\) 个柱子的高度为 \(a_i\)。每个人可以删去最左边一列或最下边一行,求谁操作后删光整个柱状图。

我们发现每次操作后,图的左下角一定会移动,并且一个左下角对应一种游戏局面。
所以,这个模型还可以转化:从左下角出发,两人轮流向上或向右走一个单位,走到边界的人输。

这是个很简单的博弈论 \(\text{dp}\),每个点对应一个操作者的必胜态 / 必败态。将操作者的胜败状态记在该操作到达的点上,则边界上全是必败态,\((1,1)\) 是后手的胜败状态(因为先手从这里出发到邻点,胜败状态在两个邻点上,通过那两个点的胜败状态算出的胜败状态 相当于后手在游戏开始时的胜败状态)。
直接 \(O(n\times a_i)\) \(\text{dp}\) 即可求出每个点是必胜态还是必败态。
但这样显然会 T,我们考虑优化。
随便画一个图,手玩出每个点的胜败状态(红圈表示必败态,蓝叉表示必胜态)

不难发现左下-右上对角线上的状态都是一样的!如何证明?
假如 \((x,y)\) 是 \(1\),\((x-1,y-1)\) 不可能是 \(0\)。这里用反证法,举一个例子:
0 ?
1 1 ?
0 1 0
注意到 \(0\) 的后继全都是 \(1\),\(1\) 的后继一定有 \(0\)。
可以画出这样的图。
发现两个 ? 处至少有一个 \(0\)(\(1\) 的后继一定有 \(0\)),但两个 ? 处都必须是 \(1\)(\(0\) 的后继全部是 \(1\)),因此矛盾。
以上图为例,我们就只需要算 \((4,4)\) 的胜败状态了!
这个位置怎么算呢?
它的上边、右边所有点都贴着边界,状态为胜、败轮流交替。于是可以从
以上图为例,从 \((4,7)\) 往下根据奇偶性推出 \((4,5)\) 的状态,从 \((8,4)\) 往左推出 \((5,4)\) 的状态,然后根据 \((4,5)\) 和 \((5,4)\) 的状态就可以推出 \((4,4)\longrightarrow (1,1)\) 的状态了。
复杂度 \(O(n)\)。
#include<bits/stdc++.h>
using namespace std;
int a[100005];
int main(){
int n,i;
scanf("%d",&n);
for(i=1; i<=n; ++i) scanf("%d",&a[i]);
sort(a+1, a+n+1, greater<int>());
for(i=1; i<=n; ++i){
if(i<=a[i] && a[i+1]<i+1){
int j=0;
while(a[j+i+1]==i) j++;
if((a[i]-i)%2==0 && j%2==0) puts("Second");
else puts("First");
return 0;
}
}
}
【AGC002 E】Candy Piles的更多相关文章
- 【AGC002E】Candy Piles 博弈论
题目大意 有\(n\)堆糖果,第\(i\)堆有\(a_i\)个. 两个人轮流决策,决策分为两种: 1.选择糖果数最多的一堆糖果,并把这堆糖全吃了. 2.在每堆非空的糖果堆里拿一颗糖吃掉. 吃掉最后一颗 ...
- 【LEETCODE OJ】Candy
Problem link: http://oj.leetcode.com/problems/candy/ Suppose we are given an array R[1..N] that are ...
- 【LeetCode练习题】Candy
分糖果 There are N children standing in a line. Each child is assigned a rating value. You are giving c ...
- 【ZOJ 3897】Candy canes//Fiddlesticks
题 题意 给你一串数,a1...an,从左到右每次让一个数减小c,如果这个数小于c,那就减为0.第n个数减小后,又从第一个开始从左到右.如果这次某个数减小到0,那就改变方向,如果遇到已经是0的,就跳过 ...
- 【HDOJ 5654】 xiaoxin and his watermelon candy(离线+树状数组)
pid=5654">[HDOJ 5654] xiaoxin and his watermelon candy(离线+树状数组) xiaoxin and his watermelon c ...
- 【Unity Shader】Unity Chan的卡通材质
写在前面 时隔两个月我终于来更新博客了,之前一直在学东西,做一些项目,感觉没什么可以分享的就一直没写.本来之前打算写云彩渲染或是Compute Shader的,觉得时间比较长所以打算先写个简单的. 今 ...
- 【赛事总结】◇赛时·8◇ AGC-027
[赛时·8]AGC-027 日常AGC坑……还好能涨Rating +传送门+ ◇ 简单总结 感觉像打多校赛一样,应该多关注一下排名……考试的时候为了避免影响心态,管都没有管排名,就在那里死坑B题.最后 ...
- 【24.67%】【codeforces 551C】 GukiZ hates Boxes
time limit per test2 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...
- DFS【搜索1】
DFS模板 void dfs(int depth)//depth表示当前的层数(或深度) { if(depth>n)//到达叶子节点,该路已走到尽头 return; for(int i=;i&l ...
随机推荐
- NDK学习笔记-C语言
本文简要回顾了C语言的一些注意事项和理解细节,不再赘述C语言的所有语法 头文件 头文件作为引入文件,在编译的时候,加载到源代码,参与编译 在VS2013中可以看到,当引入头文件时候,只能看到函数的声明 ...
- mysql CPU太高排查办法
[1]问题描述 首先,查看top,下图来自网络 为什么会有%CPU 375??? 还可以超过100%的? 这是因为,有多核CPU.如图,top后,按数字1,即可出现下图. [2]排查办法(当前CPU爆 ...
- 小记--------spark的worker原理分析及源码分析
- mongodb 批量替换内容的语句
// 注意上面的集合要和下面的集合名称相同 db.rand_web_test.find( { 'content': /^.*要替换的字符串.*$/ } ).forEach( function (ite ...
- 【问题】【编程环境】fatal error: security/pam_appl.h
[问题] 今天在docker中基于centos镜像的容器编译gogs遇到错误 似乎缺少库文件 [解决] yum -y install pam-devel
- url编码问题小计
昨天通过get访问服务器遇到了服务器获取不到参数的问题,最后排查下来是因为url编码的原因,之前使用的是字符串拼接,所以有一些特殊字符如‘%’没有正确的编码, 通过改成各个部分编码,正确获取到数据. ...
- CSS和DOM入门
CSS补充: - position - background - hover - overflow - z-index - opacity 示例:输入框右边放置图标 JavaScript: 局部变量 ...
- Java Serializable Objects(序列化)
https://docs.oracle.com/javase/8/docs/api/java/io/Serializable.html 序列化一个对象,意味着把这个对象的状态转化为字节流,而且个字节流 ...
- Scala学习四——映射和数组
一.本章要点 Scala有十分易用的语言来创建,查询和遍历映射 你需要从可变和不可变的映射中做出选择 默认情况下,你得到的是一个哈希映射,不过你也可以指明要树形映射 你可以很容易地在Scala映射和J ...
- vs code 快捷键设置:选中字母可以快速全部转换为大写或小写
文件--->首选项--->键盘快捷方式--->搜索:"大写"--->点击"转换为大写"左侧的加号,然后设置快捷键后按enter即可完成添 ...