题面传送门

题意:

  • 有一棵 \(n\) 个节点的图 \(G\),R 和 B 两个人轮流操作,R 先操作。
  • 每次操作 R 可以染红任意一条未染色的边,B 可以染蓝任意一条未染色的边
  • R 的目标是染成一棵全蓝的生成树,R 的目标是阻止后手染成一棵全蓝的生成树,问谁会赢。
  • \(n \in [1,10]\),\(m \in [1,30]\)

yet another 博弈论 problem......

不难猜出本题的结论是如果 \(G\) 中存在两棵边集不相交的生成树,答案为 \(\texttt{YES}\),否则答案为 \(\texttt{NO}\),你手玩几组数据应该就可以看出来。

但是怎么证明呢?

对于游戏中任意一个局面,我们分以下几种情况考虑:

  1. 如果 \(G\) 中存在两棵生成树 \(T_1,T_2\),满足 \(T_1,T_2\) 只由蓝边或无色边组成,并且 \(T_1,T_2\) 中无色边组成的边集不相交。

    如果 R 染红的边不在 \(T_1,T_2\) 中,那么 B 随便染蓝一条边都可以保持这个局面。因为你染蓝这条边后 \(T_1,T_2\) 依然满足上述条件。

    如果 R 染红的边在 \(T_1,T_2\) 中,不妨假设 R 染红的边为 \(e_1 \in T_1\),那么 B 必然可以找出 \(e_2 \in T_2\),满足从 \(T_1\) 中删除 \(e_1\) 加入 \(e_2\) 依然是一个生成树 \(T'_1\)。你感性理解一下,去掉 \(e_1\) 之后原树 \(T_1\) 分成两个点集 \(V_1,V_2\),由于 \(T_2\) 是一棵树,那么必然有一条边 \(e_2\) 沟通了 \(V_1,V_2\),不然 \(T_2\) 就不连通了。如果此时 \(e_2\) 未染色,那么 B 可以将 \(e_2\) 染成蓝色,此时 \(T'_1\) 与 \(T_2\) 依然满足上述条件。如果 \(e_2\) 已经染成蓝色,那么 B 随便染蓝一条边也不会使局面更坏。

    综上,这种情况下 B 有必胜策略。
  2. 如果 \(G\) 中不存在两棵生成树 \(T_1,T_2\),满足 \(T_1,T_2\) 只由蓝边或无色边组成,并且 \(T_1,T_2\) 中无色边组成的边集不相交。

    Ⅰ. 如果 \(G\) 中存在一条边 \(e\) 满足 B 将 \(e\) 染蓝后会得到 1 的局面。那么 R 可将 \(e\) 染红。此时,先手边后手,后手变先手,根据 1 的结论,R 可以染出一棵全红的生成树,B 就染不出一棵全蓝的生成树了。

    Ⅱ. 如果 \(G\) 中不存在一条边 \(e\) 满足 B 将 \(e\) 染蓝后会得到 1 的局面。那么 R 可以随便染红一条边,因为 B 不管染蓝哪条边都得不到 1 的局面。

    综上,这种情况下 R 有必胜策略。

接下来就是实现的问题。本题数据范围很小,而博主又太菜不会什么 Minmax 搜索之类的高级算法。只好采用暴搜的方法,暴力枚举一棵生成树的边集判断其他边是否能构成一棵生成树,由于 \(C_{30}^9=14307150\) 只达到 \(10^7\) 级别。因此你稍微剪点枝就可以过了。

/*
Contest: -
Problem: HDU 3267
Author: tzc_wk
Time: 2020.7.27
*/
#include <bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define fz(i,a,b) for(int i=a;i<=b;i++)
#define fd(i,a,b) for(int i=a;i>=b;i--)
#define foreach(it,v) for(__typeof(v.begin()) it=v.begin();it!=v.end();it++)
#define all(a) a.begin(),a.end()
#define fill0(a) memset(a,0,sizeof(a))
#define fill1(a) memset(a,-1,sizeof(a))
#define fillbig(a) memset(a,0x3f,sizeof(a))
#define fillsmall(a) memset(a,0xcf,sizeof(a))
#define y1 y1010101010101
#define y0 y0101010101010
typedef pair<int,int> pii;
inline int read(){
int x=0,neg=1;char c=getchar();
while(!isdigit(c)){
if(c=='-') neg=-1;
c=getchar();
}
while(isdigit(c)) x=x*10+c-'0',c=getchar();
return x*neg;
}
bool ans=0;
int n,k,m,u[75],v[75],f[75],num=0;
bool vis[75];
inline int find(int x){
return (f[x]==x)?(x):find(f[x]);
}
int to[75],ecnt=0,nxt[75],hd[75];
inline void ae(int x,int y){
to[++ecnt]=y;
nxt[ecnt]=hd[x];
hd[x]=ecnt;
}
int used[75];
inline void findcomp(int x){
if(used[x]) return;
used[x]=1;
for(int i=hd[x];i;i=nxt[i]) findcomp(to[i]);
}
inline void dfs(int x,int lst){
if(x==n){
// num++;
// if(num%100000==0) cout<<num<<endl;
// fz(i,1,m) if(vis[i]) cout<<i<<" ";puts("");
fill0(hd);fill0(nxt);fill0(to);ecnt=0;fill0(used);
fz(i,1,m) if(!vis[i]) ae(u[i],v[i]),ae(v[i],u[i]);
findcomp(1);bool flg=1;
fz(i,1,n) if(!used[i]) flg=0;
if(flg) ans=1;
return;
}
for(int i=lst+1;i<=m;i++){
int _u=u[i],_v=v[i];
// if(rand()&1) swap(_u,_v);
int fu=find(_u),fv=find(_v);
if(fu!=fv){
f[fu]=fv;vis[i]=1;dfs(x+1,i);
if(ans) return;
vis[i]=0;f[fu]=fu;f[fv]=fv;
}
}
}
inline void solve(){
m=0;fill0(u);fill0(v);fill0(f);
fz(i,1,k){
int _u=read(),_v=read();_u++;_v++;
if(_u!=_v) u[++m]=_u,v[m]=_v;
}
fz(i,1,n) f[i]=i;
fill0(vis);ans=0;dfs(1,0);
if(ans) puts("YES");
else puts("NO");
}
signed main(){
while(scanf("%d %d",&n,&k)){
if(!~n) break;
solve();
}
return 0;
}

HDU 3267 Graph Game(博弈论+图论+暴力)的更多相关文章

  1. HDU 1557 权利指数 国家压缩 暴力

    HDU 1557 权利指数 状态压缩 暴力 ACM 题目地址:HDU 1557 权利指数 题意:  中文题,不解释. 分析:  枚举全部集合,计算集合中的和,推断集合里面的团体是否为关键团队. 代码: ...

  2. [la P5031&hdu P3726] Graph and Queries

    [la P5031&hdu P3726] Graph and Queries Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: ...

  3. HDU 1524 树上无环博弈 暴力SG

    一个拓扑结构的图,给定n个棋的位置,每次可以沿边走,不能操作者输. 已经给出了拓扑图了,对于每个棋子找一遍SG最后SG和就行了. /** @Date : 2017-10-13 20:08:45 * @ ...

  4. HDU.2149 Public Sale (博弈论 巴什博弈)

    HDU.2149 Public Sale (博弈论 巴什博弈) 题意分析 巴什博奕裸题 博弈论快速入门 代码总览 #include <bits/stdc++.h> using namesp ...

  5. HDU.1846 Brave Game (博弈论 巴什博弈)

    HDU.1846 Brave Game (博弈论 巴什博弈) 题意分析 巴什博奕裸题 博弈论快速入门 代码总览 include <bits/stdc++.h> using namespac ...

  6. HDU 3726 Graph and Queries treap树

    题目来源:HDU 3726 Graph and Queries 题意:见白书 思路:刚学treap 參考白皮书 #include <cstdio> #include <cstring ...

  7. 图论/暴力 Codeforces Beta Round #94 (Div. 2 Only) B. Students and Shoelaces

    题目传送门 /* 图论/暴力:这是个连通的问题,每一次把所有度数为1的砍掉,把连接的点再砍掉,总之很神奇,不懂:) */ #include <cstdio> #include <cs ...

  8. HDU 3131 One…Two…Five! (暴力搜索)

    题目链接:pid=3131">HDU 3131 One-Two-Five! (暴力搜索) 题意:给出一串数字,要求用加,减,乘,除(5/2=2)连接(计算无优先级:5+3*6=8*6= ...

  9. HDU 4467 Graph(图论+暴力)(2012 Asia Chengdu Regional Contest)

    Description P. T. Tigris is a student currently studying graph theory. One day, when he was studying ...

随机推荐

  1. python的函数参数传递方式

    python的一切数据类型都是对象.但是python的对象分为不可变对象和可变对象.python的变量是引用,对python变量的赋值是引用去绑定该对象. 可变对象的数据发生改变,例如列表和字典,引用 ...

  2. 【UE4 C++ 基础知识】<15> 智能指针 TSharedPtr、UniquePtr、TWeakPtr、TSharedRef

    基本概念 UE4 对 UObject 对象提供垃圾回收 UE4 对原生对象不提供垃圾回收,需要手动进行清理 方式 malloc / free new / delete new与malloc的区别在于, ...

  3. 【技术博客】在Unity3d中实现烟花效果

    在游戏开发中,我们经常需要用到类似烟花的效果.在Unity3d中,实现烟花效果的方法不止一种,我选用了Unity3d中新添加的粒子特效工具--visual effect graph来进行实现. 实现过 ...

  4. 算法:N-gram语法

    一.N-gram介绍 n元语法(英语:N-gram)指文本中连续出现的n个语词.n元语法模型是基于(n - 1)阶马尔可夫链的一种概率语言模型,通过n个语词出现的概率来推断语句的结构.这一模型被广泛应 ...

  5. hdu 1856 More is better(并查集)

    题意: Mr Wang wants some boys to help him with a project. Because the project is rather complex, the m ...

  6. hdu 5057 Argestes and Sequence (数状数组+离线处理)

    题意: 给N个数.a[1]....a[N]. M种操作: S X Y:令a[X]=Y Q L R D P:查询a[L]...a[R]中满足第D位上数字为P的数的个数 数据范围: 1<=T< ...

  7. 【java设计模式】(10)---模版方法模式(案例解析)

    一.概念 1.概念 模板方法模式是一种基于继承的代码复用技术,它是一种类行为型模式. 它定义一个操作中的算法的骨架,而将一些步骤延迟到子类中.模板方法使得子类可以不改变一个算法的结构即可重定义该算法的 ...

  8. 用C++实现的数独解题程序 SudokuSolver 2.7 及实例分析

    引言:一个 bug 的发现 在 MobaXterm 上看到有内置的 Sudoku 游戏,于是拿 SudokuSolver 求解,随机出题,一上来是个 medium 级别的题: 073 000 060 ...

  9. 如何系统学习C 语言(中)之 指针篇

    谈到指针,我们可能会想到钟表上的指针,但这里的指针不是现实生活中看得见摸得着的钟表上的指针,c 语言中的指针只存在于逻辑思维中,物理上并不存在. 同时,指针也是C 语言中最精华的部分,通过灵活地运用指 ...

  10. RedHat 7.0 Linux 下划分区,分区加密,配额,逻辑卷管理

    1:如何划分区: 1:明确分区的对象:xxx :fdisk /dev/xxx 2:增加一个分区:n:选择主分区或者扩展分区,"p" or "e" :默认地方开始 ...