bzoj 2437[Noi2011]兔兔与蛋蛋 黑白染色二分图+博弈+匈牙利新姿势
noi2011 兔兔与蛋蛋
题目大意
直接看原题吧
就是\(n*m\)的格子上有一些白棋和一些黑棋和唯一一个空格
兔兔先手,蛋蛋后手
兔兔要把与空格相邻的其中一个白棋移到空格里
蛋蛋要把与空格相邻的其中一个黑棋移到空格里
谁不能移动谁输
分析
这篇博客挺好的
我们可以将题意转化成兔兔将空格移到白棋那里
蛋蛋将空格移动到黑棋那里
转化成图黑白染色,变成二分图
我们设空格染成黑色
那空格移动的轨迹一定是:
黑\(~\)-白-黑-白-黑
对应的是:
空格-白棋-黑棋-白棋-黑棋
所以染成白色且为白棋\(~\)或\(~\)染成黑色且为黑棋
的才是合法点(其他点不可能移动的)
将相邻合法点连边
同时把空格当成黑棋看
又根据奇偶性,从空格从一个点出发不可能绕回那个点
博弈
假如我们现在求出了一个最大匹配
那么如果一个点一定在最大匹配中,那么他有必胜策略
否则没有
证明:
已匹配的边记为\(A\)
未匹配的边记为\(B\)
那么从一个点出发走到无路可走\(~~\)且\(~~\)有A尽量走A有以下几种情况:(图自行脑补)
根据匹配,不可能出现\(AA\)
思考一下,不可能出现\(BB\)(因为前一个\(B\)没有匹配边\(A\)了,后一个\(B\)无路可走了)
还有就是从一个点出发有\(B\)则那个点出发还一定有个\(A\)
1.\(ABAB\)
这种情况\(AB\)可以互换,最大匹配不变
出发点不一定在最大匹配中,先手走\(A\)会让对方有必胜策略
2.\(BABA\)
同上
出发点不一定在最大匹配中,先手走\(B\)会让对方有必胜策略
3.\(BABAB\)
因为从一个点出发有\(B\)则那个点出发还一定有个\(A\)
交换\(AB\)最大匹配不变,且出发点依然在最大匹配中
出发点一定在最大匹配中,先手走\(B\)或\(A\)都能赢
4.\(ABABA\)
这种情况出发点一定在最大匹配中,先手走\(A\)就赢了
做法
证明完了
现在如何判断一个点是否一定在最大匹配中呢?
首先如果本来就不在最大匹配直接就不行了
否则删掉这个点,并断掉这个点的匹配边,从它匹配点增广
匹配点能增广它就不一定在最大匹配了
做法
于是我们动态ban点
每读入一个操作就ban掉一个点就好了
姿势
匈牙利vis数组可以用时间戳
标号都不一样的话可以两边一起匹配
bool xyl(int x){
int p,y;
for(p=g[x];p;p=e[p].nxt)
if(vis[y=e[p].y]!=T&&!del[y]){
vis[y]=T;
if(lnk[y]==-1||xyl(lnk[y])){
lnk[y]=x;//
lnk[x]=y;//
return 1;
}
}
return 0;
}
int main(){
for(i=1;i<=cnt;i++)
if(lnk[i]==-1){
T++;
xyl(i);
}
}
solution
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cctype>
#include <cmath>
#include <algorithm>
using namespace std;
const int M=1603;
const int N=43;
inline int rd(){
int x=0;bool f=1;char c=getchar();
for(;!isdigit(c);c=getchar()) if(c=='-') f=0;
for(;isdigit(c);c=getchar()) x=x*10+c-48;
return f?x:-x;
}
int n,m,nm,K,T;
int g[M],te;
struct edge{int y,nxt;}e[M<<3];
char s[N][N];
int num[N][N],cnt,ck;
bool win[2007];
int del[M];
int lnk[M];
int vis[M];
void addedge(int x,int y){
e[++te].y=y;e[te].nxt=g[x];g[x]=te;
}
bool xyl(int x){
int p,y;
for(p=g[x];p;p=e[p].nxt)
if(vis[y=e[p].y]!=T&&!del[y]){
vis[y]=T;
if(lnk[y]==-1||xyl(lnk[y])){
lnk[y]=x;
lnk[x]=y;
return 1;
}
}
return 0;
}
int main(){
int i,j,x,y,nw;
n=rd(),m=rd();
for(i=1;i<=n;i++) scanf("%s",s[i]+1);
for(i=1;i<=n;i++)
for(j=1;j<=m;j++)
if(s[i][j]=='.'){
s[i][j]='X';//要使它有连边
x=i; y=j;//x,y存的是空格位置
ck=(i+j)%2;
break;
}
for(i=1;i<=n;i++)
for(j=1;j<=m;j++){
if(s[i][j]=='X'&&(i+j)%2==ck) num[i][j]=++cnt;
if(s[i][j]=='O'&&(i+j)%2!=ck) num[i][j]=++cnt;
}
for(i=1;i<=n;i++)
for(j=1;j<=m;j++)
if(num[i][j]){
if(i>1&&num[i-1][j]) addedge(num[i][j],num[i-1][j]);
if(i<n&&num[i+1][j]) addedge(num[i][j],num[i+1][j]);
if(j>1&&num[i][j-1]) addedge(num[i][j],num[i][j-1]);
if(j<m&&num[i][j+1]) addedge(num[i][j],num[i][j+1]);
}
memset(lnk,-1,sizeof(lnk));
for(i=1;i<=cnt;i++)
if(lnk[i]==-1){
T++;
xyl(i);
}
K=rd();
for(i=1;i<=K*2;i++){
nw=num[x][y];
del[nw]=1;
if(lnk[nw]==-1) win[i]=0;
else{
int tp=lnk[nw];
lnk[nw]=lnk[tp]=-1;
T++;
win[i]= !xyl(tp);
}
x=rd(),y=rd();
}
int ans=0;
for(i=1;i<=K;i++)
if(win[i*2]&&win[i*2-1]) ans++;
printf("%d\n",ans);
for(i=1;i<=K;i++)
if(win[i*2]&&win[i*2-1]) printf("%d\n",i);
return 0;
}
bzoj 2437[Noi2011]兔兔与蛋蛋 黑白染色二分图+博弈+匈牙利新姿势的更多相关文章
- bzoj 2437: [Noi2011]兔兔与蛋蛋
Description Solution 考虑犯错误的条件:之前是处于必胜状态,该操作之后就变成了必败状态. 我们可以把这个过程看成两人对网格图进行黑白染色,变成了一个二分图模型,即当前位置向相邻不同 ...
- bzoj 2437 [Noi2011]兔子和鸡蛋 [二分图匹配]
叙述性说明 这些日子.兔子和蛋像一个新的棋盘游戏. 这场比赛是在 n 行 m 在船上进行列. 前,棋盘上有一 个格子是空的,其他的格子中都放置了一枚棋子,棋子或者是黑色,或者是白色. 每一局游戏总是兔 ...
- BZOJ.2756.[SCOI2012]奇怪的游戏(二分 黑白染色 最大流ISAP)
题目链接 \(Description\) \(Solution\) 这种题当然要黑白染色.. 两种颜色的格子数可能相同,也可能差1.记\(n1/n2\)为黑/白格子数,\(s1/s2\)为黑/白格子权 ...
- BZOJ 2131 圈地计划(最小割+黑白染色)
类似于happiness的一道题,容易想到最小割的做法. 但是不同的是那一道题是相邻的如果相同则有收益,这题是相邻的不同才有收益. 转化到建图上面时,会发现,两个相邻的点连的边容量会是负数.. 有一种 ...
- bzoj 1997: [Hnoi2010]Planar【瞎搞+黑白染色】
脑补一下给出的图:一个环,然后有若干连接环点的边,我们就是要求这些边不重叠 考虑一下不重叠的情况,两个有交边一定要一个在环内一个在环外,所以把相交的边连边,然后跑黑白染色看是否能不矛盾即可(可能算个2 ...
- BZOJ.1443.[JSOI2009]游戏Game(二分图博弈 匈牙利)
题目链接 \(Description\) 一个\(N*M\)的有障碍的棋盘,先手放置棋子后,从后手开始轮流移动棋子,不能走重复的位置,不能移动的输.求在哪些位置放棋子是先手必胜的. \(Solutio ...
- BZOJ.2437.[NOI2011]兔兔与蛋蛋游戏(二分图博弈 匈牙利)
题目链接 首先空格的移动等价于棋子在黑白格交替移动(设起点移向白格就是黑色),且不会走到到起点距离为奇数的黑格.到起点距离为偶数的白格(删掉就行了),且不会重复走一个格子. (然后策略就同上题了,只不 ...
- 【BZOJ 2437】 2437: [Noi2011]兔兔与蛋蛋 (博弈+二分图匹配**)
未经博主同意不得转载 2437: [Noi2011]兔兔与蛋蛋 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 693 Solved: 442 Des ...
- 2437: [Noi2011]兔兔与蛋蛋 - BZOJ
Description Input 输入的第一行包含两个正整数 n.m.接下来 n行描述初始棋盘.其中第i 行包含 m个字符,每个字符都是大写英文字母"X".大写英文字母" ...
随机推荐
- WINDOWS-基础:LPTSTR
1. LPTSTR解释 与char*等价,表示普通字符/字符串变量,指向字符/字符串的指针. LP: 长指针(long pointer). T: win32环境中有一个_T宏,用来标识字符是否采 ...
- “CTL_CODE”未定义
C4013 “CTL_CODE”未定义:假设外部返回 int 要加入 #include<winioctl.h> 并且要放在#include<windows.h>的后面
- HTML5语义
语义通俗化为意义,也就是语义化的元素等于意义化的元素,看到这个元素的名称,就知道这个元素的意义,是拿来做什么用的,这就是HTML5的一个新特性,一个具有语义化的元素能够清楚的把元素的意义告诉浏览器和开 ...
- mutt+msmtp做linux邮件客户端
mutt+msmtp做linux邮件客户端 1. 安装配置msmtp l 安装 wget https://sourceforge.net/projects/msmtp/files/msmtp/1.4 ...
- ANSI C 与 K&R C
C语言由Dennis M.Ritchie在1973年设计和实现.从那以后使用者逐渐增加.到1978年Ritchie和Bell实验室的另一位程序专家Kernighan合写了著名的<TheC Pro ...
- NOIP2016——大家一起实现の物语
由于最近硬盘挂了,换了个固态硬盘,比赛结束后四天一直在装Linux,所以最近一直没怎么更新 看起来挺漂亮的 比赛前一个月申请停了一个月晚自习,在我们这座城市里能做到这种事情已经可以被称为奇迹了,并且在 ...
- 洛谷 2023 [AHOI2009]维护序列
洛谷 2023 [AHOI2009]维护序列 洛谷原题传送门 这个题也是一道经典的线段树模版(其实洛谷的模版二改一下输入顺序就能AC),其中包括区间乘法修改.区间加法修改.区间查询三个操作. 线段树的 ...
- mysql命令行复制数据库
为了方便快速复制一个数据库,可以用以下命令将db1数据库的数据以及表结构复制到newdb数据库创建新的数据库#mysql -u root -p123456 mysql>CREATE DATABA ...
- destoon模块自定义字段的添加并让其支持搜索的方法
今天看了看模块设置里的自定义字段功能的用法,试着加了个新字段glry,设置了值,然后去数据库moduleid的article表看,字段成功加上了. 于是去template下article文件夹的lis ...
- ProC第三弹
一.前言 我们上面已经了解Windows和Linux下的ProC开发环境,这里我们更进一步去简要介绍下ProC的预编译参数. 二.什么是预编译 预编译过程中,Pro*C/C++会自动生成C或者C++的 ...