zoukankan      html  css  js  c++  java
  • LOJ6033「雅礼集训 2017 Day2」棋盘游戏 (博弈论,二分图,匈牙利算法)

    什么神仙思路啊……

    看到棋盘就去想二分图。(smg啊)(其实是校内模拟赛有基本一样的题,只不过直接给了个二分图)

    看到二分图就去想最大匹配。(我怎么想偶环的性质去了)

    (以下内容摘自这里

    这个二分图的某种最大匹配方案中,从非匹配点出发先手必败:先手只能走到匹配点(否则不是最大匹配),后手只需要一直走匹配边即可,先手操作时不可能走到非匹配点(否则存在增广路,与最大匹配矛盾),所以先手必败。

    容易发现,当且仅当出发点一定在最大匹配中,先手才会胜利。

    (注:这里我觉得有点问题,虽然我大概能感受到究竟为什么是对的,但我说不出,所以咕了)

    所以,一个起点使先手必胜当且仅当它一定在最大匹配中。

    判断的话,先求出任意一个最大匹配。然后对有匹配的点,如果与他有匹配的点能找到另一个匹配,那么它实际上不合法,否则合法。

    用匈牙利算法即可做到 (O((nm)^2))。实际上常数小,跑不满,跑得飞快。

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=100010,mod=998244353,d[4][2]={{0,1},{0,-1},{1,0},{-1,0}};
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline int read(){
    	int x=0,f=0;char ch=getchar();
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int n,m,el,cnt,head[maxn],to[maxn],nxt[maxn],dis[maxn],with[maxn],x[maxn],y[maxn],id[111][111],tot;
    char mp[111][111];
    bool vis[maxn],ans[maxn];
    inline void add(int u,int v){
    	to[++el]=v;nxt[el]=head[u];head[u]=el;
    }
    void dfs(int u){
    	for(int i=head[u];i;i=nxt[i]){
    		int v=to[i];
    		if(~dis[v]) continue;
    		dis[v]=dis[u]^1;
    		dfs(v);
    	}
    }
    bool dfs2(int u){
    	if(vis[u]) return false;
    	vis[u]=true;
    	for(int i=head[u];i;i=nxt[i]){
    		int v=to[i];
    		if(!with[v] || dfs2(with[v])){
    			with[with[u]]=0;
    			with[u]=v;with[v]=u;
    			return true;
    		}
    	}
    	return false;
    }
    int main(){
    	n=read();m=read();
    	FOR(i,1,n) scanf("%s",mp[i]+1);
    	FOR(i,1,n) FOR(j,1,m) if(mp[i][j]=='.') id[i][j]=++cnt,x[cnt]=i,y[cnt]=j;
    	FOR(i,1,n) FOR(j,1,m) if(mp[i][j]=='.'){
    		FOR(k,0,3){
    			int ti=i+d[k][0],tj=j+d[k][1];
    			if(ti<1 || ti>n || tj<1 || tj>m || mp[ti][tj]=='#') continue;
    			add(id[i][j],id[ti][tj]);
    		}
    	}
    	MEM(dis,-1);
    	FOR(i,1,cnt) if(dis[i]==-1) dis[i]=0,dfs(i);
    	FOR(i,1,cnt) if(dis[i]){
    		MEM(vis,0);
    		dfs2(i);
    	}
    	FOR(i,1,cnt) if(with[i]){
    		MEM(vis,0);
    		ans[i]=!dfs2(with[i]);
    	}
    	FOR(i,1,cnt) if(!ans[i]) tot++;
    	printf("%d
    ",tot);
    	FOR(i,1,cnt) if(!ans[i]) printf("%d %d
    ",x[i],y[i]);
    }
    
  • 相关阅读:
    oracle本地编译问题
    ORA-214 signalled during: ALTER DATABASE MOUNT 问题
    mysql常用的一些修改命令
    了解PHP中的Array数组和foreach
    在Sublime Text 3上安装代码格式化插件CodeFormatter
    自动在图片上添加页码
    Ubuntu 16.10 安装byzanz截取动态效果图工具
    PHP赋值运算
    Ubuntu 16.10 安装KolourPaint 4画图工具
    PHP数据类型之间的强制转换
  • 原文地址:https://www.cnblogs.com/1000Suns/p/11806024.html
Copyright © 2011-2022 走看看