POJ 1753 Flip Game (高斯消元)

来源:互联网 发布:java oa办公系统 开源 编辑:程序博客网 时间:2024/06/06 12:41

题目地址:POJ 1753

第三次做这道题了。第一次是刚学搜索的时候做的,第二次是刚学状态压缩枚举的时候做的,这次是刚学高斯消元、、每次都做得很艰辛。。目测这题应该没了别的方法了吧。。。。。。

这题除了高斯消元外还需要枚举变元,方法是状态压缩枚举,然后返回去迭代解方程。

代码如下:

#include <iostream>#include <string.h>#include <math.h>#include <queue>#include <algorithm>#include <stdlib.h>#include <map>#include <set>#include <stdio.h>using namespace std;#define LL __int64#define pi acos(-1.0)const int mod=1e9+7;const int INF=0x3f3f3f3f;const double eqs=1e-6;int mp[5][5], a[20][20], free_num, free_x[20], x[20];int jx[]={0,0,1,-1};int jy[]={1,-1,0,0};int gauss(){        int i, j, k, free_num=0, h, tmp, t;        for(i=0,j=0;i<16&&j<16;i++,j++){                if(a[i][j]==0){                        for(k=i+1;k<16;k++){                                if(a[k][j]) break;                        }                        if(k==16){                                free_x[free_num++]=j;                                i--;                                continue ;                        }                        for(h=j;h<17;h++){                                swap(a[i][h],a[k][h]);                        }                }                for(k=i+1;k<16;k++){                        if(a[k][j]){                                for(h=j;h<17;h++){                                        a[k][h]^=a[i][h];                                }                        }                }        }        tmp=i;        for(j=i;j<16;j++){                if(a[j][16]) return INF;        }        int ans=INF;        int tot=1<<free_num;        //printf("%d\n",free_num);        for(i=0;i<tot;i++){                int cnt=0;                for(j=0;j<free_num;j++){                        if(i&(1<<j)){                                x[free_x[j]]=1;                                cnt++;                        }                        else{                                x[free_x[j]]=0;                        }                }                //printf("%d\n",cnt);                for(j=tmp-1;j>=0;j--){                        t=0;                        while(a[j][t]==0) t++;                        x[t]=a[j][16];                        for(k=t+1;k<16;k++){                                if(a[j][k])  x[t]^=x[k];                        }                        cnt+=x[t];                }                //printf("%d\n",cnt);                ans=min(ans,cnt);        }        return ans;}void init(){        memset(a,0,sizeof(a));        for(int i=0;i<4;i++){                for(int j=0;j<4;j++){                        for(int k=0;k<4;k++){                                int x=i+jx[k];                                int y=j+jy[k];                                if(x>=0&&x<4&&y>=0&&y<4){                                        a[4*i+j][4*x+y]=1;                                }                        }                        a[4*i+j][4*i+j]=1;                }        }}int main(){        char s[6];        int i, j, k, x, y, ans1, ans2;        for(i=0;i<4;i++){                scanf("%s",s);                for(j=0;j<4;j++){                        if(s[j]=='b') mp[i][j]=0;                        else mp[i][j]=1;                }        }        init();        for(i=0;i<4;i++){                for(j=0;j<4;j++){                        a[4*i+j][4*4]=mp[i][j];                }        }        ans1=gauss();        init();        for(i=0;i<4;i++){                for(j=0;j<4;j++){                        a[4*i+j][4*4]=1-mp[i][j];                }        }        ans2=gauss();        if(ans1==INF&&ans2==INF){                printf("Impossible\n");        }        else{                printf("%d\n",min(ans1,ans2));        }        return 0;}


1 0
原创粉丝点击