题意:略。

思路:这是个删边的博弈游戏。

关于删边游戏的预备知识:http://blog.csdn.net/acm_cxlove/article/details/7854532

学习完预备知识后,这一题就不难了。

首先,用tarjan算法找到每棵树中的双连通分量(即树枝上的多边形),根据Fusion Principle,如果多边形有奇数条边,可以简化为1条边,如果有偶数条边,则可以简化为1个点。代码中使用了vis数组,对于前者,使环内所有的点(包括悬挂点)的vis值为1,后面计算sg值时便不会再进行遍历这些点;对于后者,除了悬挂点和多边形内与悬挂点相邻的一点(相邻点中只取一个)外,多边形内其他点vis为1,这样就相当于将环化为了1条边。另外,题目当中会有重边,根据题意,当出现这种情况时,一定也是出现在树枝的末端。这里直接当作多边形来处理,方法是相同的,如果有偶数条边,化为1个点,奇数条边化为1条边。

然后由Colon Principle,可以计算出整棵树的sg值,之后对于所有的树,就是个Nim游戏了。

更多细节看代码

 #include<stdio.h>
#include<string.h>
#include<stack>
#include<algorithm>
#define maxn 105
#define maxp 2000
using namespace std;
int map[maxn][maxn];
bool vis[maxn];
struct node
{
int v, next;
}edge[maxp];
int num_edge, head[maxn];
void init()
{
num_edge = ;
memset(head, -, sizeof(head));
}
void addedge(int a,int b)
{
edge[num_edge].v = b;
edge[num_edge].next = head[a];
head[a] = num_edge++;
}
struct scc
{
int dfn[maxn], low[maxn];
int cnt, scnt;
bool instack[maxn];
stack<int> s;
void init()
{
cnt = scnt = ;
memset(dfn, -, sizeof(dfn));
memset(instack, , sizeof(instack));
}
void tarjan(int u,int father)
{
dfn[u] = low[u] = ++cnt;
s.push(u);
instack[u] = ;
for (int i = head[u]; i != -; i = edge[i].next)
{
int v = edge[i].v;
if (v == father)
{
if (map[u][v] > && map[u][v] % == )
vis[u] = ;
continue;
}
if (dfn[v] == -)
{
tarjan(v, u);
low[u] = min(low[u], low[v]);
}
else if (instack[v]) low[u] = min(low[u], dfn[v]);
}
if (dfn[u] == low[u])
{
scnt = ;
int last;
for (;;)
{
int temv = s.top();
s.pop();
scnt++;
instack[temv] = ;
if ((temv == u) || s.empty())
break;
vis[temv] = ;
last = temv;
}
if (scnt & )
vis[last] = ;
}
return;
}
void solve(int n)
{
init();
for (int i = ; i <= n; i++) if (dfn[i] == -)
tarjan(i, -);
}
};
int getsg(int u,int father)
{
int osum = ;
for (int i = head[u]; i != -; i = edge[i].next)
{
int v = edge[i].v;
if (!vis[v] && v != father)
osum ^= ( + getsg(v, u));
}
return osum;
}
int main()
{
int k, n, m;
scc g;
while (~scanf("%d", &n))
{
int osum = ;
for (int i = ; i <= n; i++)
{
scanf("%d%d", &m, &k);
int a, b;
init();
memset(map, , sizeof(map));
memset(vis, , sizeof(vis));
while (k--)
{
scanf("%d%d", &a, &b);
addedge(a, b);
map[a][b]++;
addedge(b, a);
map[b][a]++;
}
g.solve(m);
osum ^= getsg(, -);
}
if (osum) printf("Sally\n");
else printf("Harry\n");
}
return ;
}

POJ 3710 Christmas Game [博弈]的更多相关文章

  1. POJ 3710 Christmas Game#经典图SG博弈

    http://poj.org/problem?id=3710 (说实话对于Tarjan算法在搞图论的时候就没搞太懂,以后得找时间深入了解) (以下有关无向图删边游戏的资料来自论文贾志豪<组合游戏 ...

  2. poj 3710 Christmas Game(树上的删边游戏)

    Christmas Game Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 1967   Accepted: 613 Des ...

  3. poj 3710 Christmas Game 博弈论

    思路:首先用Tarjan算法找出树中的环,环为奇数变为边,为偶数变为点. 之后用博弈论的知识:某点的SG值等于子节点+1后的异或和. 代码如下: #include<iostream> #i ...

  4. POJ 3710 Christmas Game

    知识储备: 解决办法(奇偶去环):   (1) 对于长度为奇数的环,去掉其中任意一个边之后,剩下的 两个链长度同奇偶,抑或之后的 SG 值不可能为奇数,所 以它的 SG 值为 1: (2) 对于长度为 ...

  5. POJ.3710.Christmas Game(博弈论 树上删边游戏 Multi-SG)

    题目链接 \(Description\) 给定n棵"树",每棵"树"的节点可能"挂着"一个环,保证没有环相交,且与树只有一个公共点. 两人轮 ...

  6. poj 3710 Christmas Game【博弈论+SG】

    也就是转换到树形删边游戏,详见 https://wenku.baidu.com/view/25540742a8956bec0975e3a8.html #include<iostream> ...

  7. POJ 3710 无向图简单环树上删边

    结论题,这题关键在于如何转换环,可以用tarjan求出连通分量后再进行标记,也可以DFS直接找到环后把点的SG值变掉就行了 /** @Date : 2017-10-23 19:47:47 * @Fil ...

  8. Solution -「POJ 3710」Christmas Game

    \(\mathcal{Decription}\)   Link.   定义一棵圣诞树: 是仙人掌. 不存在两个同一环上的点,度数均 \(\ge 3\).   给出 \(n\) 棵互不相关的圣诞树,双人 ...

  9. poj 2975 Nim(博弈)

    Nim Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 5232   Accepted: 2444 Description N ...

随机推荐

  1. Android Shader渲染器:BitmapShader图片渲染

    public class BitmapShader extends Shader BitmapShader,  Shader家族的 专门处理图片渲染的 构造方法: public BitmapShade ...

  2. BZOJ 4971: [Lydsy1708月赛]记忆中的背包

    神仙构造 分成x个1和一堆>=w-x的大物品 (x<=20 w>=50) 则拼成w的方案中有且仅有一个大物品 若最终序列中有x个1,有一个大物品为w-k,可以提供C(x,k)种方案 ...

  3. 54、edittext输入类型限制为ip,inputType应该如何设置

    <EditText android:id="@+id/et_setting_printer_edit_info_ip" android:layout_width=" ...

  4. 【Luogu P1120】小木棍

    题目: 乔治有一些同样长的小木棍,他把这些木棍随意砍成几段,直到每段的长都不超过$50$.现在,他想把小木棍拼接成原来的样子,但是却忘记了自己开始时有多少根木棍和它们的长度.给出每段小木棍的长度,编程 ...

  5. 查看sqlserver表空间

    exec sp_msforeachtable "exec sp_spaceused '?'"

  6. Solr安装 win系统

    安装之前需查看:https://lucene.apache.org/solr/guide/7_6/solr-system-requirements.html#solr-system-requireme ...

  7. zookeeper 下载安装

    下载:wget https://www-us.apache.org/dist/zookeeper/zookeeper-3.4.13/zookeeper-3.4.13.tar.gz 解压:tar -zx ...

  8. easyui loader 改变rows total page rows等参数名称!

    公司需要对接客户接口,但客户接口已经确定,分页请求的参数以及返回的数据是客户自定义的名称,与easyui 所封装的参数无法对应,这是需要改变参数名称,这时我们可以使用loader方法: loader: ...

  9. mysql-Innodb事务隔离级别-repeatable read详解

    http://blog.csdn.net/dong976209075/article/details/8802778 经验总结: Python使用MySQLdb数据库后,如使用多线程,每个线程创建一个 ...

  10. 流浪者(rover)

    流浪者(rover) 题目描述 有一位流浪者正在一个n∗mn∗m的网格图上流浪.初始时流浪者拥有SS点体力值. 流浪者会从(1,1)(1,1)走向(n,m)(n,m),并且他只会向下走((x,y)→( ...