题目

分析

第一次做这种题,其实很简单。

只能经过一次的博弈可以考虑转化为二分图博弈。

棋盘上有黑白色的棋子,可以把这个游戏看作空格在棋子间移动,于是就想到,把棋盘黑白染色,以空格为黑,那么空格的移动轨迹一定是黑白相间的。发现有一些棋子空格是移不过去的,那就是染色与棋子颜色不同的点(由于兔兔白棋先走,所以把空格染成黑色,可以符合要求)。剩下的点,把它们分成两部分,黑色和白色,那么可以组成一个二分图,所有的路径其实就是二分图上的路径,因为不可能有两个白色或黑色的棋子相邻。这个问题转化成了一个二分图上的博弈问题,不能走到重复的点。

对于二分图上的博弈,有一个结论:必胜点必定在最大匹配上。也就是说,如果有多种匹配情况,而一个点在某些情况中在最大匹配中,而在其他情况中不在,那么这个点不是必胜的。

下面说明这个结论:

如果一个点\(x\)一定在最大匹配中,那么当前操作者沿着这个点的匹配边走出去,走到对面的匹配点\(y\)。点\(x\)已经不能再走了,但由于\(x\)必定在最大匹配中,这就说明,删除点\(x\)后点\(y\)找不到一条新的增广路,所以\(y\)能走到的任意下一条边一定是非匹配边,连接着一个匹配点。所以轮到先手时,总是可以在一个匹配点上,它的匹配边连出去的另一个点没有被走过。由于边不可能有无限多,而先手总是有办法走,所以后手一定会输。

所以这题直接记录每一个走到的点是否一定在新图的最大匹配中,如果兔兔走之前在,之后也在,那么她就走错了。注意要把走过的删掉。

代码

#include<cstdio>
#include<cctype>
#include<algorithm>
#include<cstring>
using namespace std;
int read() {
int x=0,f=1;
char c=getchar();
for (;!isdigit(c);c=getchar()) if (c=='-') f=-1;
for (;isdigit(c);c=getchar()) x=x*10+c-'0';
return x*f;
}
const int xx[]={-1,0,1,0};
const int yy[]={0,1,0,-1};
const int maxn=51;
const int maxm=maxn*maxn;
char s[maxn][maxn];
bool a[maxn][maxn],alr[maxm],cannot[maxm],able[maxm];
int white[maxm];
int sx,sy,n,m,wrong[maxm],wt=0,match[maxm],all;
struct egde {
int v,nxt;
} e[maxm<<3];
int h[maxm],tot=0;
void add(int u,int v) {
e[++tot]=(egde){v,h[u]};
h[u]=tot;
}
int dis(int ax,int ay,int bx,int by) {
return abs(ax-bx)+abs(ay-by);
}
int ID(int x,int y) {
return (x-1)*m+y;
}
int change(int x) {
return x&1?x+1:x-1;
}
bool dfs(int x) {
for (int i=h[x],v=e[i].v;i;i=e[i].nxt,v=e[i].v) if (!alr[v] && able[v]) {
alr[v]=true;
if (!match[v] || dfs(match[v])) {
match[v]=x;
match[x]=v;
return true;
}
}
return false;
}
bool win[maxm];
int main() {
#ifndef ONLINE_JUDGE
freopen("test.in","r",stdin);
#endif
n=read(),m=read();
all=ID(n,m);
for (int i=1;i<=n;++i) {
scanf("%s",s[i]+1);
for (int j=1;j<=m;++j) if (s[i][j]=='.') sx=i,sy=j;
}
for (int i=1;i<=n;++i) for (int j=1;j<=m;++j) {
int id=ID(i,j);
white[id]=dis(i,j,sx,sy)&1;
if ((s[i][j]=='O' && white[id]) || ((s[i][j]=='X' || s[i][j]=='.') && !white[id])) able[id]=true; else able[id]=false;
}
for (int i=1;i<=n;++i) for (int j=1;j<=m;++j) {
int aid=ID(i,j);
if (!able[aid]) continue;
for (int k=0;k<4;++k) {
int x=xx[k]+i,y=yy[k]+j;
if (x<1 || x>n || y<1 || y>m) continue;
int bid=ID(x,y);
if (able[bid]) {
add(bid,aid),add(aid,bid);
}
}
}
for (int i=1;i<=all;++i) if (!white[i]) {
memset(alr,0,sizeof alr);
dfs(i);
}
int k=read();
for (int i=1;i<=(k<<1);++i) {
int id=ID(sx,sy);
able[id]=false;
if (match[id]) {
int tmp=match[id];
match[id]=match[tmp]=0;
memset(alr,0,sizeof alr);
win[i]=!dfs(tmp);
} else win[i]=false;
sx=read(),sy=read();
}
wt=0;
for (int i=1;i<=k;++i) if (win[i*2-1] && win[i*2]) ++wt;
printf("%d\n",wt);
for (int i=1;i<=k;++i) if (win[i*2-1] && win[i*2]) printf("%d\n",i);
return 0;
}

bzoj2437-兔兔与蛋蛋的更多相关文章

  1. 【BZOJ2437】【NOI2011】兔兔与蛋蛋(博弈论,二分图匹配)

    [BZOJ2437][NOI2011]兔兔与蛋蛋(博弈论,二分图匹配) 题面 BZOJ 题解 考虑一下暴力吧. 对于每个状态,无非就是要考虑它是否是必胜状态 这个直接用\(dfs\)爆搜即可. 这样子 ...

  2. 【bzoj2437】[Noi2011]兔兔与蛋蛋 二分图最大匹配+博弈论

    Description Input 输入的第一行包含两个正整数 n.m. 接下来 n行描述初始棋盘.其中第i 行包含 m个字符,每个字符都是大写英文字母"X".大写英文字母&quo ...

  3. 2437: [Noi2011]兔兔与蛋蛋 - BZOJ

    Description Input 输入的第一行包含两个正整数 n.m.接下来 n行描述初始棋盘.其中第i 行包含 m个字符,每个字符都是大写英文字母"X".大写英文字母" ...

  4. 博弈论(二分图匹配):NOI 2011 兔兔与蛋蛋游戏

    Description Input 输入的第一行包含两个正整数 n.m. 接下来 n行描述初始棋盘.其中第i 行包含 m个字符,每个字符都是大写英文字母"X".大写英文字母&quo ...

  5. NOI2011 兔兔与蛋蛋游戏

    http://www.lydsy.com/JudgeOnline/problem.php?id=2437 这道题真是极好的. 75分做法: 搜索. 出题人真的挺良心的,前15个数据点的范围都很小,可以 ...

  6. 【BZOJ 2437】 2437: [Noi2011]兔兔与蛋蛋 (博弈+二分图匹配**)

    未经博主同意不得转载 2437: [Noi2011]兔兔与蛋蛋 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 693  Solved: 442 Des ...

  7. bzoj 2437[Noi2011]兔兔与蛋蛋 黑白染色二分图+博弈+匈牙利新姿势

    noi2011 兔兔与蛋蛋 题目大意 直接看原题吧 就是\(n*m\)的格子上有一些白棋和一些黑棋和唯一一个空格 兔兔先手,蛋蛋后手 兔兔要把与空格相邻的其中一个白棋移到空格里 蛋蛋要把与空格相邻的其 ...

  8. BZOJ2437 [Noi2011]兔兔与蛋蛋 【博弈论 + 二分图匹配】

    题目链接 BZOJ2437 题解 和JSOI2014很像 只不过这题动态删点 如果我们把空位置看做\(X\)的话,就会发现我们走的路径是一个\(OX\)交错的路径 然后将图二分染色,当前点必胜,当且仅 ...

  9. BZOJ2437 NOI2011兔兔与蛋蛋(二分图匹配+博弈)

    首先将棋盘黑白染色,不妨令空格处为黑色.那么移动奇数次后空格一定处于白色格子,偶数次后空格一定处于黑色格子.所以若有某个格子的棋子颜色与棋盘颜色不同,这个棋子就是没有用的.并且空格与某棋子交换后,棋子 ...

  10. bzoj2437 [Noi2011]兔兔与蛋蛋

    二分图博弈果然都是一个套路,必经点必胜,非必经点必败, 但是肯定不能每走一步就重新建图判断必胜还是必败,那么我们可以这样:每走一步就把这个点删掉,然后find他原来的匹配,如果找不到,就说明他是必经点 ...

随机推荐

  1. Android Studio 引入 Git 并提交代码

    File -> Settings -> Version Control -> Git -> Path to Git executable -> 选择本地 Git 可执行文 ...

  2. 数据库 MySQL part3

    外键约束 如果表A的主关键字是表B中的字段,则该字段称为表B的外键,表A称为主表,表B称为从表. 外键是用来实现参照完整性的,不同的外键约束方式将可以使两张表紧密的结合起来,特别是修改或者删除的级联操 ...

  3. See You Again——我最后的汇编程序

    汇编语言:课程设计2 前言 由于本人水平不够,这里的课程设计2的程序实现并没有像王爽书中所说的那样可以不依赖于操作系统运行. 这里的程序依然要在dos下运行,而且没有实现引导现有操作系统的功能. 该程 ...

  4. c#调用c++库函数

    如果是非托管的,就用DllImport,举例    using System;    using System.Runtime.InteropServices;    class MainApp    ...

  5. 【BZOJ4818】序列计数(动态规划,生成函数)

    [BZOJ4818]序列计数(生成函数) 题面 BZOJ 题解 显然是求一个多项式的若干次方,并且是循环卷积 或者说他是一个\(dp\)也没有问题 发现项数很少,直接暴力乘就行了(\(FFT\)可能还 ...

  6. vs2015 mvc项目数据迁移报错

    第一次做个mvc项目玩玩,然后需要数据迁移,也没做过,就百度找怎么数据迁移, 找到的方法是: 如果数据是在类库项目里就在‘程序包管理控制台’输入:enable-migrations -ContextT ...

  7. Oracle锁表处理

    最近系统连续出现好几次锁表,昨晚又发生一次锁表,11点钟跑到客户现场,进过跟踪发现导致这次锁表的机器和上一次是同一台,花了近半小时解锁.之后到科室找到那台机器看看情况,发现那台机器速度超慢,保存一份病 ...

  8. 第一阶段·Linux运维基础 第3章·文件属性、正则表达式、文件权限

    01-文件属性内容介绍 02- inodeyublock讲解 03-访问oldboyfile的寻宝过程 04-inode与block小结 05-磁盘空间不足案例详解 06-Linux文件类型及拓展名 ...

  9. 拥抱移动端,jQueryui触控设备兼容插件

    http://touchpunch.furf.com/ ps:要FQ. jQuery UI Touch Punch Touch Event Support for jQuery UI Tested o ...

  10. python常用命令—‘\r’

    # \r 默认表示将输出的内容返回到第一个指针,这样的话,后面的内容会覆盖前面的内容 如常用的显示程序完成进度!!