【BZOJ】【3164】【HEOI2013】Eden的博弈问题

树形DP



  这题在考场上直接写的TreeDP……当时也没想出一个像样的暴力来对拍……好像只能这么直接做了……?

  都说是博弈树了,转移关系都给的这么直接了……也没啥难度了吧= =(怪不得大家都不愿意写题解)

  我的思路是这样的:

  如果黑方想赢,那么:

  1.在一个决策方为黑方的节点,对于它来说,最小黑方胜集合就是所有儿子中最小的那个 最小黑方胜集合

  2.在一个决策方为白方的节点,它的最小黑方胜集合是所有儿子的最小黑方胜集合的并

  那么现在我们就可以通过子节点的最小黑方胜集合,推出在某个节点状态下的最小黑方胜集合的大小,然而我们容易发现:哪些元素可能是最小黑方胜集合中的元素,也是满足这个转移的:

  1.对每个节点维护一个mn[x]表示这个最小xx集合的大小,如果mn[son[i]]==mn[fa],就把son的节点集合并入fa的,如果mn[son[i]]<mn[fa],就把fa的节点集合改为son的。

  第二种情况下的话直接并吧……$mn[fa]=\sum mn[son[i]]$

同理我们也能找出所有可能在最小白方胜集合中的叶节点……

求个并,随便搞搞出解即可

P.S.这题其实求树上某个节点的“最小黑方胜”集合内可能的节点时,求集合的并其实最简单的,拿链表搞搞就可以……但是蒟蒻不会写QAQ(没写过),加上昨天刚好考了一道可并堆的题目,就拿可并堆写了,权当复习……

  1 /**************************************************************
  2     Problem: 3164
  3     User: Tunix
  4     Language: C++
  5     Result: Accepted
  6     Time:648 ms
  7     Memory:30304 kb
  8 ****************************************************************/
  9
 10 //Huce #6 B
 11 #include<vector>
 12 #include<cstdio>
 13 #include<cstdlib>
 14 #include<cstring>
 15 #include<iostream>
 16 #include<algorithm>
 17 #define rep(i,n) for(int i=0;i<n;++i)
 18 #define F(i,j,n) for(int i=j;i<=n;++i)
 19 #define D(i,j,n) for(int i=j;i>=n;--i)
 20 using namespace std;
 21
 22 int getint(){
 23     int v=0,sign=1; char ch=getchar();
 24     while(ch<‘0‘||ch>‘9‘) {if (ch==‘-‘) sign=-1; ch=getchar();}
 25     while(ch>=‘0‘&&ch<=‘9‘) {v=v*10+ch-‘0‘; ch=getchar();}
 26     return v*sign;
 27 }
 28 typedef long long LL;
 29 const int N=400010,INF=~0u>>2;
 30 /*******************tamplate********************/
 31 int hd[N],to[N],ne[N],cnt;
 32 void add(int x,int y){
 33     to[++cnt]=y; ne[cnt]=hd[x]; hd[x]=cnt;
 34 }
 35 struct node{int v,l,r,dis;}t[N];
 36 #define L t[x].l
 37 #define R t[x].r
 38 int n,m,black[N],white[N],rt[N],mn[N],tot,fa[N];
 39
 40 int merge(int x,int y){
 41     if (!x || !y) return x+y;
 42     if (t[x].v>t[y].v) swap(x,y);
 43     R=merge(R,y);
 44     if (t[L].dis<t[R].dis) swap(L,R);
 45     t[x].dis=t[R].dis+1;
 46     return x;
 47 }
 48 bool sign[N];
 49 void dfs(int x,bool now){
 50     if (!hd[x]){
 51         mn[x]=1;
 52         rt[x]=++tot;
 53         t[tot].v=x;
 54         t[tot].l=t[tot].r=t[tot].dis=0;
 55         return;
 56     }
 57     if (sign[x]==now) mn[x]=INF; else mn[x]=0;
 58     for(int i=hd[x];i;i=ne[i]){
 59         sign[to[i]]=sign[x]^1;
 60         dfs(to[i],now);
 61         if (sign[x]==now){
 62             if (mn[x]>mn[to[i]]){
 63                 mn[x]=mn[to[i]];
 64                 rt[x]=rt[to[i]];
 65             }else if (mn[x]==mn[to[i]]){
 66                 rt[x]=merge(rt[x],rt[to[i]]);
 67             }
 68         }else{
 69             mn[x]+=mn[to[i]];
 70             rt[x]=merge(rt[x],rt[to[i]]);
 71         }
 72     }
 73 }
 74 int ans[N];
 75 int main(){
 76 #ifndef ONLINE_JUDGE
 77     freopen("B.in","r",stdin);
 78 //  freopen("output.txt","w",stdout);
 79 #endif
 80     n=getint();
 81     F(i,2,n){
 82         fa[i]=getint(); add(fa[i],i);
 83     }
 84     tot=0; sign[1]=1;
 85     dfs(1,1);
 86     int num1=0;
 87     while(rt[1]){black[++num1]=t[rt[1]].v; rt[1]=merge(t[rt[1]].l,t[rt[1]].r);}
 88     sort(black+1,black+num1+1);
 89     tot=0; memset(rt,0,sizeof rt);
 90     dfs(1,0);
 91     int num2=0;
 92     while(rt[1]){white[++num2]=t[rt[1]].v; rt[1]=merge(t[rt[1]].l,t[rt[1]].r);}
 93     sort(white+1,white+num2+1);
 94     int num=0;
 95     for(int i=1,j=1;i<=num1 && j<=num2;i++){
 96         while(black[i]>white[j] && j<=num2) j++;
 97         if (black[i]==white[j]) ans[++num]=black[i];
 98     }
 99     int sum=0;
100     F(i,1,num) sum^=ans[i];
101     printf("%d %d %d\n",ans[1],num,sum);
102     return 0;
103 }
104 

3164: [Heoi2013]Eden的博弈问题

Time Limit: 10 Sec  Memory Limit: 256 MB
Submit: 110  Solved: 84
[Submit][Status][Discuss]

Description

对于有两个玩家的,状态透明且状态转移确定的博弈游戏,博弈树是常用的分析工具。博弈树是一棵有根树,其中的节点为游戏的状态。若节点B的父亲是A,则说
明状态A能通过一次决策转移到状态B。每个状态都有一个唯一的决策方,即这个状态下应该由哪一方做出决策。我们规定双方在任何时候都是轮流做出决策的,即
树上相邻节点的决策方总是不相同的。在这个问题中,我们只关心两个玩家的胜负情况,且规定游戏不会出现平局。
我们称两个玩家分别为黑方和白方,其中根节点的决策方为黑方。显然每个节点 只有两个状态:黑方胜和白方胜。若某内节点(即存在后继节点的节点)的决策
方为黑方,则该节点为黑方胜的充要条件为它的儿子中存在黑方胜的节点,反之亦然。求解博弈树即为判明博弈树根节点的状态。如果我们得知了所有叶节点(即无
后继节点的节点)的状态,那么博弈树就
很容易求解了。但是现在的情况是所有叶节点的状态均为未知的,需要进一步的计算。对于一个由叶节点构成的集合S,如果S中的节点均被判明为黑方胜,就可以
断言根节点为黑方胜的话,则称 S为一个黑方胜集合。对于黑方胜集合 S,
如果对于任意的黑方胜集合 S’均满足|S| ≤ |S’ |(|S|表示集合S中的元素数目),
 则称S为一个最小黑方胜集合。同样地,也可以定义白方胜集合和最小白方胜集合。
 Eden最近在研究博弈树问题。他发现,如果一个叶节点既属于某一个最小黑方胜集合,又属于一个最小白方胜集合,那么求解这个节点的状态显然最有益
于求解根节点的状态。像这样的叶节点就称之为关键叶节点。对于一棵给定的博弈树,Eden想要知道哪些叶节点是关键叶节点。

Input

每个测试点包含一组测试数据。
测试数据的第一行包含一个正整数n,表示博弈树的节点数目。节点从1到n 编号,且 1 号节点为根节点。
之后n–1 行,每行包含一个正整数。第i行的正整数表示节点i的父节点的编号。

Output

在一行内输出三个空格分隔的正整数,分别是编号最小的关键叶节点的编号,
关键叶节点的数目和所有关键叶节点的编号的异或和。

Sample Input

7
1
1
2
2
3
3

Sample Output

4 4 0

HINT

对于100% 的数据,1 ≤  n ≤ 200,000 ,且对于节点 i(i ≠ 1 ),其父节点的编号小于i。

Source

[Submit][Status][Discuss]

时间: 2025-01-13 19:28:28

【BZOJ】【3164】【HEOI2013】Eden的博弈问题的相关文章

BZOJ 3163 Heoi2013 Eden的新背包问题 多重背包

题目大意:多重背包,多次询问某个物品不能选择时以某个总价钱最多能获得多少价值 求问正解是啥QAQ 维护一个前缀多重背包和一个后缀多重背包 每次询问时 枚举前面选多少和后面选多少 暴力统计答案即可 时间复杂度O(n^2logn+nq) 这3E的复杂度居然只跑了600sQAQ 正解到底是啥QAQ #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> #defin

BZOJ3163&amp;Codevs1886: [Heoi2013]Eden的新背包问题[分治优化dp]

3163: [Heoi2013]Eden的新背包问题 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 428  Solved: 277[Submit][Status][Discuss] Description “寄没有地址的信,这样的情绪有种距离,你放着谁的歌曲,是怎样的心心静,能不能说给我听.”失忆的Eden总想努力地回忆起过去,然而总是只能清晰地记得那种思念的感觉,却不能回忆起她的音容笑貌. 记忆中,她总是喜欢给Eden出谜题:在 valent

bzoj3163: [Heoi2013]Eden的新背包问题

Description “寄没有地址的信,这样的情绪有种距离,你放着谁的歌曲,是怎样的心心静,能不能说给我听.”失忆的Eden总想努力地回忆起过去,然而总是只能清晰地记得那种思念的感觉,却不能回忆起她的音容笑貌. 记忆中,她总是喜欢给Eden出谜题:在 valentine’s day 的夜晚,两人在闹市中闲逛时,望着礼品店里精巧玲珑的各式玩偶,她突发奇想,问了 Eden这样的一个问题:有n个玩偶,每个玩偶有对应的价值.价钱,每个玩偶都可以被买有限次,在携带的价钱m固定的情况下,如何选择买哪些玩偶

BZOJ 3166: [Heoi2013]Alo

3166: [Heoi2013]Alo Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 923  Solved: 437[Submit][Status][Discuss] Description Welcome to ALO ( Arithmetic and Logistic Online).这是一个VR MMORPG ,如名字所见,到处充满了数学的谜题.现在你拥有n颗宝石,每颗宝石有一个能量密度,记为ai,这些宝石的能量密度两两不同.现在你可以选

BZOJ 3165: [Heoi2013]Segment

3165: [Heoi2013]Segment Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 465  Solved: 187[Submit][Status][Discuss] Description 要求在平面直角坐标系下维护两个操作: 1.在平面上加入一条线段.记第i条被插入的线段的标号为i. 2.给定一个数k,询问与直线 x = k相交的线段中,交点最靠上的线段的编号. Input 第一行一个整数n,表示共n 个操作. 接下来n行,每行第一

Bzoj 3165 [Heoi2013]Segment题解

3165: [Heoi2013]Segment Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 668  Solved: 276[Submit][Status][Discuss] Description 要求在平面直角坐标系下维护两个操作: 1.在平面上加入一条线段.记第i条被插入的线段的标号为i. 2.给定一个数k,询问与直线 x = k相交的线段中,交点最靠上的线段的编号. Input 第一行一个整数n,表示共n 个操作. 接下来n行,每行第一

BZOJ 1299 [LLH邀请赛]巧克力棒 博弈(NIM游戏)+构造

题意:链接 方法:博弈+构造 解析: 神题没想到咋做,看完wyf大爷说的话才懂 这道题的题意就是有n堆石子,之后你每次操作有两种做法 第一种是在已经建立的nim游戏上进行nim游戏. 第二种是在未被选的堆中选取若干堆加入这个nim游戏中. 所以这怎么做? 我一直尝试去建立一个必败态,不过可能我走的方法过多的考虑了将这道题转变为nim游戏套nim游戏而最终走入死胡同,不会做了. 然而这道题的必败态可以这么建:如果是先手的话,我可以建立出来一个异或和为0的nim游戏,此时后者有两种做法,第一种是在该

bzoj 3165: [Heoi2013]Segment 动态凸壳

3165: [Heoi2013]Segment Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 202  Solved: 89[Submit][Status] Description 要求在平面直角坐标系下维护两个操作: 1.在平面上加入一条线段.记第i条被插入的线段的标号为i.   2.给定一个数k,询问与直线 x = k相交的线段中,交点最靠上的线段的编号. Input 第一行一个整数n,表示共n 个操作. 接下来n行,每行第一个数为0或1.

[HEOI2013]Eden 的新背包问题

题目描述 " 寄 没 有 地 址 的 信 ,这 样 的 情 绪 有 种 距 离 ,你 放 着 谁 的 歌 曲 ,是 怎 样 的 心 情 . 能 不 能 说 给 我 听 ." 失忆的 Eden 总想努力地回忆起过去,然而总是只能清晰地记得那种思念的 感觉,却不能回忆起她的音容笑貌. 记忆中,她总是喜欢给 Eden 出谜题:在 valentine's day 的夜晚,两人在闹市 中闲逛时,望着礼品店里精巧玲珑的各式玩偶,她突发奇想,问了 Eden 这样的 一个问题:有 n 个玩偶,每个玩偶