【bzoj2437】[Noi2011]兔兔与蛋蛋 二分图最大匹配+博弈论
Description
Input
输入的第一行包含两个正整数 n、m。
接下来 n行描述初始棋盘。其中第i 行包含 m个字符,每个字符都是大写英文字母"X"、大写英文字母"O"或点号"."之一,分别表示对应的棋盘格中有黑色棋子、有白色棋子和没有棋子。其中点号"."恰好出现一次。
接下来一行包含一个整数 k(1≤k≤1000) ,表示兔兔和蛋蛋各进行了k次操作。
接下来 2k行描述一局游戏的过程。其中第 2i – 1行是兔兔的第 i 次操作(编号为i的操作) ,第2i行是蛋蛋的第i次操作。每个操作使用两个整数x,y来描述,表示将第x行第y列中的棋子移进空格中。
输入保证整个棋盘中只有一个格子没有棋子, 游戏过程中兔兔和蛋蛋的每个操作都是合法的,且最后蛋蛋获胜。
Output
输出文件的第一行包含一个整数r,表示兔兔犯错误的总次数。
接下来r 行按递增的顺序给出兔兔“犯错误”的操作编号。其中第 i 行包含一个整数ai表示兔兔第i 个犯错误的操作是他在游戏中的第 ai次操作。
1 ≤n≤ 40, 1 ≤m≤ 40
Sample Input
样例一:
1 6
XO.OXO
1
1 2
1 1
样例二:
3 3
XOX
O.O
XOX
4
2 3
1 3
1 2
1 1
2 1
3 1
3 2
3 3
样例三:
4 4
OOXX
OXXO
OO.O
XXXO
2
3 2
2 2
1 2
1 3
Sample Output
样例一:
1
1
样例二:
0
样例三:
2
1
2
样例1对应图一中的游戏过程
样例2对应图三中的游戏过程
Sol
最近补了几发博弈论,但都是和sg函数有关的博弈,看到这题就懵了。
后来才知道这个叫二分图博弈......
具体地,这个操作相当于在移动这个空格,我们把空格变成黑色,那么就是按照黑白黑白黑白的顺序走,这好像有点二分图的感觉...
然后我们把合法的点拉出来建四联通的二分图(到起点的曼哈顿距离和它的颜色正好匹配),之后这个博弈其实就是在二分图上跑增广路,而且经过的边不能再次经过,这样的话我们发现,一个点是必胜态当且仅当这个点一定是一个最大匹配包含的点,因为匹配边和连接匹配的边总个数是个奇数。。。只要一直顺着匹配连边走就能赢。。。(对手走的是连接两对匹配的边)否则的话一定是必败态(因为走一步一定走到了一个最大匹配上,假设走不到说明还有新的匹配,与最大匹配矛盾,不成立)。
至于判断一个点是不是最大匹配的必需点,我们把它删掉,从它的原匹配点如果能找到最短路,说明它不是必需点。判断有没有做错是需要判断两个连续的状态是不是都是必胜态。。。
然后好像就完了......
Code
#include <bits/stdc++.h>
using namespace std;
char s[45][45];vector<int>e[2500];
int n,m,q,tot,Index,ans,a[45][45],sx,sy,u,v,vis[2500],match[2500],del[2500],win[2500];
bool dfs(int x)
{
for(int i=0;i<e[x].size();i++)
{
if(del[e[x][i]]||vis[e[x][i]]==Index) continue;
vis[e[x][i]]=Index;
if(!match[e[x][i]]||dfs(match[e[x][i]])){match[e[x][i]]=x,match[x]=e[x][i];return 1;}
}
return 0;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++) scanf("%s",s[i]+1);
for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) if(s[i][j]=='.'){sx=i,sy=j,s[i][j]='X';break;}
for(int i=1;i<=n;i++) for(int j=1;j<=m;j++)
if((s[i][j]=='O'&&(abs(i-sx)+abs(j-sy))%2==1)||(s[i][j]=='X'&&(abs(i-sx)+abs(j-sy))%2==0)) a[i][j]=++tot;
for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) if(a[i][j])
{
if(a[i-1][j]) e[a[i][j]].push_back(a[i-1][j]);if(a[i+1][j]) e[a[i][j]].push_back(a[i+1][j]);
if(a[i][j-1]) e[a[i][j]].push_back(a[i][j-1]);if(a[i][j+1]) e[a[i][j]].push_back(a[i][j+1]);
}
for(int i=1;i<=tot;i++) if(!match[i]) Index++,dfs(i);
scanf("%d",&q);q<<=1;
for(int i=1;i<=q;i++)
{
u=a[sx][sy];
if(match[u]) v=match[u],match[u]=match[v]=0,del[u]=1,Index++,win[i]=!dfs(v);
else del[u]=1,win[i]=0;
scanf("%d%d",&sx,&sy);
}
for(int i=1;i<=q;i++,i++) if(win[i]&&win[i+1]) ans++;
printf("%d\n",ans);
for(int i=1;i<=q;i++,i++) if(win[i]&&win[i+1]) printf("%d\n",(i+1)>>1);
}
【bzoj2437】[Noi2011]兔兔与蛋蛋 二分图最大匹配+博弈论的更多相关文章
- 【BZOJ2437】【NOI2011】兔兔与蛋蛋(博弈论,二分图匹配)
[BZOJ2437][NOI2011]兔兔与蛋蛋(博弈论,二分图匹配) 题面 BZOJ 题解 考虑一下暴力吧. 对于每个状态,无非就是要考虑它是否是必胜状态 这个直接用\(dfs\)爆搜即可. 这样子 ...
- 【BZOJ 2437】 2437: [Noi2011]兔兔与蛋蛋 (博弈+二分图匹配**)
未经博主同意不得转载 2437: [Noi2011]兔兔与蛋蛋 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 693 Solved: 442 Des ...
- bzoj 2437[Noi2011]兔兔与蛋蛋 黑白染色二分图+博弈+匈牙利新姿势
noi2011 兔兔与蛋蛋 题目大意 直接看原题吧 就是\(n*m\)的格子上有一些白棋和一些黑棋和唯一一个空格 兔兔先手,蛋蛋后手 兔兔要把与空格相邻的其中一个白棋移到空格里 蛋蛋要把与空格相邻的其 ...
- BZOJ.2437.[NOI2011]兔兔与蛋蛋游戏(二分图博弈 匈牙利)
题目链接 首先空格的移动等价于棋子在黑白格交替移动(设起点移向白格就是黑色),且不会走到到起点距离为奇数的黑格.到起点距离为偶数的白格(删掉就行了),且不会重复走一个格子. (然后策略就同上题了,只不 ...
- 2437: [Noi2011]兔兔与蛋蛋 - BZOJ
Description Input 输入的第一行包含两个正整数 n.m.接下来 n行描述初始棋盘.其中第i 行包含 m个字符,每个字符都是大写英文字母"X".大写英文字母" ...
- 博弈论(二分图匹配):NOI 2011 兔兔与蛋蛋游戏
Description Input 输入的第一行包含两个正整数 n.m. 接下来 n行描述初始棋盘.其中第i 行包含 m个字符,每个字符都是大写英文字母"X".大写英文字母&quo ...
- NOI2011 兔兔与蛋蛋游戏
http://www.lydsy.com/JudgeOnline/problem.php?id=2437 这道题真是极好的. 75分做法: 搜索. 出题人真的挺良心的,前15个数据点的范围都很小,可以 ...
- 【BZOJ2432】【NOI2011】兔农(数论,矩阵快速幂)
[BZOJ2432][NOI2011]兔农(数论,矩阵快速幂) 题面 BZOJ 题解 这题\(75\)分就是送的,我什么都不想写. 先手玩一下,发现每次每次出现\(mod\ K=1\)的数之后 把它减 ...
- POJ 2226二分图最大匹配
匈牙利算法是由匈牙利数学家Edmonds于1965年提出,因而得名.匈牙利算法是基于Hall定理中充分性证明的思想,它是二部图匹配最常见的算法,该算法的核心就是寻找增广路径,它是一种用增广路径求二分图 ...
随机推荐
- ndarray的创建与数据类型
ndarray 多维数组(N Dimension Array) NumPy数组是一个多维的数组对象(矩阵),称为ndarray,具有矢量算术运算能力和复杂的广播能力,并具有执行速度快和节省空间的特点. ...
- JAVA中跨平台分隔符
在Windows下的路径分隔符和Linux下的路径分隔符是不一样的,当直接使用绝对路径时,跨平台会暴出“No such file or diretory”的异常. 比如说要在temp目录下建立一个te ...
- iTunes 安装ipa文件到iPhone上
iTunes 安装ipa文件到iPhone上 把ipa文件拖到itunes里面. ipa(iPhoneApplication) 菜单 File>Add File to Library>Se ...
- 斯坦福CS229机器学习课程笔记 Part1:线性回归 Linear Regression
机器学习三要素 机器学习的三要素为:模型.策略.算法. 模型:就是所要学习的条件概率分布或决策函数.线性回归模型 策略:按照什么样的准则学习或选择最优的模型.最小化均方误差,即所谓的 least-sq ...
- jQuery+css模拟下拉框模糊搜索的实现
html: @*输入框*@ <div> <input type="text" style="width: 85%; height: 34px;" ...
- c++多线程编程(二)
这是道面试题目:有三个线程分别打印A.B.C,请用多线程编程实现,在屏幕上循环打印10次ABCABC… 见代码: #include <iostream> #include <Wind ...
- JS事件冒泡和事件捕获的详解
在学校,听老师讲解事件冒泡和事件捕获机制的时候跟听天书一样,只依稀记得IE使用的是事件冒泡,其他浏览器则是事件捕获.当时的我,把它当成IE浏览器兼容问题,所以没有深究(IE8以下版本的浏览器已基本退出 ...
- 2014蓝桥杯B组初赛试题《奇怪的分式》
题目描述: 上小学的时候,小明经常自己发明新算法.一次,老师出的题目是: 1/4 乘以 8/5 小明居然把分子拼接在一起,分母拼接在一起,答案是:18/45 (参见图1.png) ...
- logback 中文手册
摘自:http://aub.iteye.com/blog/1896611 logback 中文手册 博客分类: Log loglogbackloback手册loback中文手册 logback 常 ...
- 手打的table
突然觉得,如果我不上传源码和写篇博客,对不起花在这个破网页2个小时的时间,完全手打,浏览器调效果. 源码如下: a.html: <!DOCTYPE html PUBLIC "-//W3 ...