Description

  

   有一棵\(n\)个节点的树(\(n \le 3000\)),第\(i\)条边连接\(a_i,b_i\),每个节点\(i\)上有\(A_i\)个石子,高桥君和青木君将在树上玩游戏。

  

​   首先,高桥君会选一个节点并在上面放一个棋子,然后从高桥君开始,他们轮流执行以下操作:

  

​   (1)从当前棋子占据的点上移除一个石子;

  

​   (2)将棋子移动到相邻节点

  

​   如果轮到一个人执行操作时棋子占据的点上没有石子,那么他就输了 。

  

​   请你找出所有的点\(v\),使得如果高桥君在游戏开始时把棋子放到\(v\)上,他可以赢。(按编号从小到大输出)

  

  

  

Solution

  

​   首先两个人的行动是互相约束的。

  

​   假设当前在节点\(u\),先手能耗死后手(即先手必胜)当且仅当对于其所有相邻点,至少存在一个点\(v\),满足:

  

​   (1)\(a_u>a_v\)

  

​   (2)\(v\)先手必败。

  

​   首先(1)是这题对局的一种博弈过程,设想有且只有两个点\(u\)和\(v\),若初始时棋子在\(u\),且\(a_u>a_v\),那么反复走,后手必死。

  

​   因此只要先手走向了如是的\(v\),后手必定不会在这条边上反复横跳,之后也不会,因为一旦后手走回来,先手继续走回\(v\),可以把后手耗到死。

  

​   那么后手必定也只能在\(v\)中寻找机会。只要\(v\)是先手必败态,那么\(u\)即为先手必胜态,因为先手可以主动走到\(v\)引出必败态。

  

​  

  

​   定义\(u\)是先手必败态当且仅当不存在上述\(v\)。

  

​   首先,如果先手走向的点\(v\)满足\(a_u\le a_v\),后手可以走回\(u\),因为反复横跳后先手必死。因此这些点不可走。

  

​   走向的点\(v\)满足\(a_u>a_v\)时,若\(v\)为先手必胜态,那么\(u\)肯定不能走这一步;如果不存在\(v\)是先手必败态,那么先手就无路可走了。

  

​   综上,因为必须走一步,所以\(u\)是先手必败态,当且仅当不存在\(v\)满足\(a_u>a_v\)且\(v\)先手必败。

  

​   对于每一个点,以其为根深搜,设\(f_u\)表示\(u\)是必胜还是必败,自底向上DP一遍。

  

​  为什么可以自底向上单向考虑?我们是要DP判定每个点\(u\)是不是必胜态,即要找到是否存在相邻点\(v\)满足\(a_u>a_v\),并深搜计算它们的必胜必败态。而对于不满足条件的\(v\),我们甚至不需要递归进去计算,因为先手不会选择走这边。所以,会选择\(u\)的父亲作为\(v\)吗?不会。每递归到\(u\)时,也就意味着,是上一步的先手想逼我(当前先手)反复横跳才走这一步过来,即满足\(a_{fa}>a_u\),所以当前先手肯定不能走父亲回去和他反复横跳。因此可以说,DP过程中,是一路向下,走后继递归计算的。

  

   时间复杂度\(\mathcal O(n^2)\)。

  

   我真TM可以退役了。

Code

#include <cstdio>
using namespace std;
const int N=3005;
int n,a[N],f[N];
int h[N],tot;
struct Edge{int v,next;}e[N*2];
inline void addEdge(int u,int v){
e[++tot]=(Edge){v,h[u]}; h[u]=tot;
e[++tot]=(Edge){u,h[v]}; h[v]=tot;
}
void readData(){
scanf("%d",&n);
for(int i=1;i<=n;i++) scanf("%d",a+i);
int u,v;
for(int i=1;i<n;i++){
scanf("%d%d",&u,&v);
addEdge(u,v);
}
}
void dfs(int u,int fa){
f[u]=0;
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa&&a[u]>a[v]){
dfs(v,u);
if(f[v]==0){
f[u]=1;
return;
}
}
}
void solve(){
for(int u=1;u<=n;u++){
dfs(u,0);
if(f[u])
printf("%d ",u);
}
}
int main(){
readData();
solve();
return 0;
}

【AGC010F】Tree Game的更多相关文章

  1. 【AGC010F】Tree Game 博弈论+暴力

    Description ​ 有一棵nn个节点的树,第ii条边连接ai,biai,bi,每个节点ii上有AiAi个石子,高桥君和青木君将在树上玩游戏 ​ 首先,高桥君会选一个节点并在上面放一个棋子,然后 ...

  2. 【POJ3237】Tree 树链剖分+线段树

    [POJ3237]Tree Description You are given a tree with N nodes. The tree's nodes are numbered 1 through ...

  3. 【BZOJ】【2631】Tree

    LCT 又一道名字叫做Tree的题目…… 看到删边加边什么的……又是动态树问题……果断再次搬出LCT. 这题比起上道[3282]tree的难点在于需要像线段树维护区间那样,进行树上路径的权值修改&am ...

  4. 【Luogu1501】Tree(Link-Cut Tree)

    [Luogu1501]Tree(Link-Cut Tree) 题面 洛谷 题解 \(LCT\)版子题 看到了顺手敲一下而已 注意一下,别乘爆了 #include<iostream> #in ...

  5. 【BZOJ3282】Tree (Link-Cut Tree)

    [BZOJ3282]Tree (Link-Cut Tree) 题面 BZOJ权限题呀,良心luogu上有 题解 Link-Cut Tree班子提 最近因为NOIP考炸了 学科也炸了 时间显然没有 以后 ...

  6. 【AtCoder3611】Tree MST(点分治,最小生成树)

    [AtCoder3611]Tree MST(点分治,最小生成树) 题面 AtCoder 洛谷 给定一棵\(n\)个节点的树,现有有一张完全图,两点\(x,y\)之间的边长为\(w[x]+w[y]+di ...

  7. 【HDU5909】Tree Cutting(FWT)

    [HDU5909]Tree Cutting(FWT) 题面 vjudge 题目大意: 给你一棵\(n\)个节点的树,每个节点都有一个小于\(m\)的权值 定义一棵子树的权值为所有节点的异或和,问权值为 ...

  8. 【BZOJ2654】Tree(凸优化,最小生成树)

    [BZOJ2654]Tree(凸优化,最小生成树) 题面 BZOJ 洛谷 题解 这道题目是之前\(Apio\)的时候写的,忽然发现自己忘记发博客了... 这个万一就是一个凸优化, 给所有白边二分一个额 ...

  9. 【POJ1741】Tree(点分治)

    [POJ1741]Tree(点分治) 题面 Vjudge 题目大意: 求树中距离小于\(K\)的点对的数量 题解 完全不觉得点分治了.. 简直\(GG\),更别说动态点分治了... 于是来复习一下. ...

随机推荐

  1. 2019网易笔试题C++--丰收

    题目描述 又到了丰收的季节,恰好小易去牛牛的果园里游玩. 牛牛常说他多整个果园的每个地方都了如指掌,小易不太相信,所以他想考考牛牛. 在果园里有N堆苹果,每堆苹果的数量为ai,小易希望知道从左往右数第 ...

  2. django orm 操作表

    django orm 操作表 1.基本操作 增 models.Tb1.objects.create(c1='xx', c2='oo') 增加一条数据,可以接受字典类型数据 **kwargs inser ...

  3. HDU 1556 Color the ball (一维树状数组,区间更新,单点查询)

    中文题,题意就不说了 一开始接触树状数组时,只知道“单点更新,区间求和”的功能,没想到还有“区间更新,单点查询”的作用. 树状数组有两种用途(以一维树状数组举例): 1.单点更新,区间查询(即求和) ...

  4. mysql/mybatis之合并两个表的查询结果

    下面这段sql是把两个表中各自符合条件的count值相加,返回结果是两个之和 SELECT sum(result) FROM ( SELECT COUNT(*) result FROM TEST_A ...

  5. 第二阶段Sprint冲刺会议9

    进展:查看有关“共享平台”的资料,看如何实现上传下载功能,并尝试编码,没有成功.

  6. Alpha版会议总结

    目前的进度: 实现了文字备忘的录入: 实现了提醒功能: 实现了可视化界面: 语音录入功能还没有完成: 界面相当粗糙: 遇到的问题: 语音录入按钮按下后没有反应,目前没有解决思路和方法. 原本的解屏功能 ...

  7. 我现在对Git的认识

    由于时间关系,我还没能真正的了解什么是Git,只是大致的了解了一下,并且在网上查阅了资料,做了一些总结,以便进一步研读. Git是一款免费.开源的分布式版本控制系统,用于敏捷高效地处理任何或小或大的项 ...

  8. Maven3 用Maven创建第一个web项目(2)servlet演示

    上一章用Maven新建了web项目成功后,本文演示在此基础上应用servlet. 1.首先修改pom.xml文件,添加servlet依赖 <project xmlns="http:// ...

  9. pktgen-dpdk 运行 run.py 报错 Config file 'default' not found 解决方法

    pktgen 操作手册:http://pktgen-dpdk.readthedocs.io/en/latest/getting_started.html 执行到这一步时: $ cd <Pktge ...

  10. windows服务外壳工具

    Windows下将nginx安装为服务运行       今天看到nginx这个小服务器软件正式版更新到了1.4.2,想玩下它.这个服务器软件虽小,但功能强大,是开源软件,有着良好的性能,被很多个人.企 ...