本来实在写不动这题 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的更多相关文章

  1. 【AGC002E】Candy Piles 博弈论

    题目大意 有\(n\)堆糖果,第\(i\)堆有\(a_i\)个. 两个人轮流决策,决策分为两种: 1.选择糖果数最多的一堆糖果,并把这堆糖全吃了. 2.在每堆非空的糖果堆里拿一颗糖吃掉. 吃掉最后一颗 ...

  2. 【LEETCODE OJ】Candy

    Problem link: http://oj.leetcode.com/problems/candy/ Suppose we are given an array R[1..N] that are ...

  3. 【LeetCode练习题】Candy

    分糖果 There are N children standing in a line. Each child is assigned a rating value. You are giving c ...

  4. 【ZOJ 3897】Candy canes//Fiddlesticks

    题 题意 给你一串数,a1...an,从左到右每次让一个数减小c,如果这个数小于c,那就减为0.第n个数减小后,又从第一个开始从左到右.如果这次某个数减小到0,那就改变方向,如果遇到已经是0的,就跳过 ...

  5. 【HDOJ 5654】 xiaoxin and his watermelon candy(离线+树状数组)

    pid=5654">[HDOJ 5654] xiaoxin and his watermelon candy(离线+树状数组) xiaoxin and his watermelon c ...

  6. 【Unity Shader】Unity Chan的卡通材质

    写在前面 时隔两个月我终于来更新博客了,之前一直在学东西,做一些项目,感觉没什么可以分享的就一直没写.本来之前打算写云彩渲染或是Compute Shader的,觉得时间比较长所以打算先写个简单的. 今 ...

  7. 【赛事总结】◇赛时·8◇ AGC-027

    [赛时·8]AGC-027 日常AGC坑……还好能涨Rating +传送门+ ◇ 简单总结 感觉像打多校赛一样,应该多关注一下排名……考试的时候为了避免影响心态,管都没有管排名,就在那里死坑B题.最后 ...

  8. 【24.67%】【codeforces 551C】 GukiZ hates Boxes

    time limit per test2 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...

  9. DFS【搜索1】

    DFS模板 void dfs(int depth)//depth表示当前的层数(或深度) { if(depth>n)//到达叶子节点,该路已走到尽头 return; for(int i=;i&l ...

随机推荐

  1. 关于字符串中每个单词的首字母大写化问题之 拆分split(/\s+/)

    var a = 'Hi, my name\'s Han Meimei, a SOFTWARE engineer'; //for循环 function titleCase(s) { var i, ss  ...

  2. ZOJ Problem Set - 1005

    注意,条件:B>=C .应考虑B=C的情况. #include<iostream> using namespace std; int A,B,C; void jugs(int a,i ...

  3. Stream系列(十四)parallet方法使用

    并发 视频讲解:https://www.bilibili.com/video/av78408286/ EmployeeTestCase.java package com.example.demo; i ...

  4. 数据检索grep

    linux操作中,总是会输出很多的内容.但是有些内容并不是我们重点关注的,所以为了看起来方便,也为了提升效率,就将不需要的内容过滤掉.  只输出想要的东西. grep: 用于搜索 模式参数(给定的字符 ...

  5. Redis 是怎么实现 “附近的人” 的?

    针对"附近的人"这一位置服务领域的应用场景,常见的可使用PG.MySQL和MongoDB等多种DB的空间索引进行实现. 而Redis另辟蹊径,结合其有序队列zset以及geohas ...

  6. 文件操作:w,w+,r,r+,a,wb,rb

    1.文件操作是什么? 操作文件: f = open("文件路径",mode="模式",encoding="编码") open() # 调用操 ...

  7. python新手必躺的5大坑

    python新手必躺的5大坑 对于Python新手来说,写代码很少考虑代码的效率和简洁性,因此容易造成代码冗长.执行慢,这些都是需要改进的地方.本文是想通过几个案列给新手一点启发,怎样写python代 ...

  8. 异或运算符(^)、与运算符(&)、或运算符(|)、反运算符(~)、右移运算符(>>)、无符号右移运算符(>>>)

    目录 异或(^).异或和 的性质及应用总结 异或的含义 异或的性质:满足交换律和结合律 异或的应用 按位 与运算符(&) 按位 或运算符(|) 取 反运算符(~) 右移运算符(>> ...

  9. Java Web 拦截器和过滤器的区别

    一.AOP:面向切面编程,Java Web中有两个常用的技术:拦截器.过滤器 二.拦截器 1.定义:在某个方法或字段被访问之前,进行拦截然后在之前或之后加入某些操作 2.原理:大部分时候,拦截器方法都 ...

  10. js中数组的定义方法及注意事项(转)

    1.数组的创建 var name= new Array(); //创建一个数组 name[0]="zhangsan";   //给数组赋值 name[1]="lisi&q ...