题目地址: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;
}