• 【LOJ】#2447. 「NOI2011」兔兔与蛋蛋的游戏


    题解

    对于75分来说,操作肯定不会成环,可以暴搜
    看成空格在移动,空格移动到原来的位置肯定经历了偶数个格子,但是操作的人是两个不同的人,所以肯定不会成环

    对于满分做法,要找到一种更好的方式判先手是否会胜

    我们看成空格在移动,每次空格必然是走一个黑棋,走一个白棋,这显然是一条交错路,我们考虑二分图

    把先手看成白色,后手看成黑色,因为空格可以移动到先手所在的位置,所以空格看成黑色

    在每个相邻的黑白格子之间连一条边

    我们就相当于在这条路上找一条路,起点是空格,使得经过的的边数是奇数,后手每个决策的点都只能走出偶数条边

    我们考虑最大匹配

    如果一个点不一定在最大匹配上
    那么这个点一定会顺着匹配边走出一条偶数条边的交错路(这时候这偶数条边在匹配内的状态全部取反,这个点就不在最大匹配内了)

    如果一个点一定在最大匹配上
    那么这个点一定不会走出一条偶数条边的交错路

    如果一个点不一定在最大匹配上,那么这个点的相邻点一定在最大匹配上
    如果这两个点都不在某个最大匹配上,那么这两个点可以匹配

    如果这个点在某个最大匹配上,相邻点不在匹配中,由于这个点不一定在最大匹配上,可以走出一条偶数条边的交错路,加上到相邻点的这条边,是可以增广的,就不是最大匹配了

    那么我们可以发现,如果起点一定在最大匹配上,那么先手必胜,如果起点不一定在最大匹配上,它下一次移动到任意一个点都是后手必胜

    所以如果空格所在的点必定在最大匹配上,先手必胜,否则,后手必胜

    这就变成了跑2000次二分图匹配的题了

    代码

    #include <iostream>
    #include <algorithm>
    #include <cstdio>
    #include <cstring>
    #include <vector>
    #include <set>
    #include <cmath>
    #include <bitset>
    #define enter putchar('
    ')
    #define space putchar(' ')
    //#define ivorysi
    #define pb push_back
    #define mo 974711
    #define pii pair<int,int>
    #define mp make_pair
    #define fi first
    #define se second
    #define MAXN 1000005
    using namespace std;
    typedef long long int64;
    typedef double db;
    template<class T>
    void read(T &res) {
        res = 0;char c = getchar();T f = 1;
        while(c < '0' || c > '9') {
    	if(c == '-') f = -1;
    	c = getchar();
        }
        while(c >= '0' && c <= '9') {
    	res = res * 10 - '0' + c;
    	c = getchar();
        }
        res = res * f;
    }
    template<class T>
    void out(T x) {
        if(x < 0) {x = -x;putchar('-');}
        if(x >= 10) out(x / 10);
        putchar('0' + x % 10);
    }
    int N,M,K,posx,posy;
    int ans[1005],tot;
    char s[45][45];
    int matk[2005];
    int dx[] = {0,-1,0,1},dy[] = {1,0,-1,0};
    bool vis[2005];
    struct node {
        int to,next;
    }E[8005];
    int head[2005],sumE;
    void add(int u,int v) {
        E[++sumE].to = v;
        E[sumE].next = head[u];
        head[u] = sumE;
    }
    int id(int x,int y) {
        return (x - 1) * M + y;
    }
    bool match(int x) {
        for(int i = head[x] ; i ; i = E[i].next) {
    	int v = E[i].to;
    	if(!vis[v]) {
    	    vis[v] = 1;
    	    if(!matk[v] || match(matk[v])) {
    		matk[v] = x;
    		return true;
    	    }
    	}
        }
        return false;
    }
    bool check(char c) {
        memset(head,0,sizeof(head));
        memset(matk,0,sizeof(matk));
        sumE = 0;
        for(int i = 1 ; i <= N ; ++i) {
    	for(int j = 1 ; j <= M ; ++j) {
    	    if(s[i][j] == c) {
    		for(int k = 0 ; k < 4 ; ++k) {
    		    int tx = i + dx[k],ty = j + dy[k];
    		    if(tx < 1 || tx > N || ty < 1 || ty > M) continue;
    		    if(s[tx][ty] != c) {
    			add(id(i,j),id(tx,ty));
    			add(id(tx,ty),id(i,j));
    		    }
    		}
    	    }
    	}
        }
        for(int i = 1 ; i <= N ; ++i) {
    	for(int j = 1 ; j <= M ; ++j) {
    	    if(s[i][j] == c) {
    		memset(vis,0,sizeof(vis));
    		match(id(i,j));
    	    }
    	}
        }
        if(!matk[id(posx,posy)]) return false;
        memset(vis,0,sizeof(vis));
        vis[id(posx,posy)] = 1;
        if(!match(matk[id(posx,posy)])) return true;
        else return false;
    }
    void Solve() {
        read(N);read(M);
        for(int i = 1 ; i <= N ; ++i) {
    	scanf("%s",s[i] + 1);
    	for(int j = 1 ; j <= M ; ++j) {
    	    if(s[i][j] == '.') {
    		posx = i;posy = j;
    	    }
    	}
        }
        read(K);
        int x,y;
        for(int i = 1 ; i <= K ; ++i) {
    	read(x);read(y);
    	bool flag1 = check('O');
    	swap(s[x][y],s[posx][posy]);
    	posx = x;posy = y;
    	
    	bool flag2 = check('X');
    	if(flag1 && flag2) ans[++tot] = i;
    	read(x);read(y);
    	swap(s[x][y],s[posx][posy]);
    	posx = x;posy = y;
        }
        out(tot);enter;
        for(int i = 1 ; i <= tot ; ++i) {
    	out(ans[i]);enter;
        }
    }
    int main() {
    #ifdef ivorysi
        freopen("f1.in","r",stdin);
    #endif
        Solve();
    }
    
  • 相关阅读:
    [CF750D] New Year and Fireworks(暴力,规律,dfs)
    [HDOJ6034] Balala Power!(模拟,贪心,大数)
    [HIHO1560] H国的身份证号码II(dp,计数,矩阵快速幂)
    [HIHO1555] 四次方根(递推,容斥,矩阵快速幂)
    [HIHO1554] 最短的 Nore0061(DP)
    [CF831D] Office Keys(二分,贪心)
    [HDOJ6156] Palindrome Function(数位dp, 枚举)
    团体程序设计天梯赛-练习集-L1-029. 是不是太胖了
    团体程序设计天梯赛-练习集-L1-028. 判断素数
    团体程序设计天梯赛-练习集-L1-027. 出租
  • 原文地址:https://www.cnblogs.com/ivorysi/p/9197981.html
Copyright © 2020-2023  润新知