题目地址: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; }
原文地址:http://blog.csdn.net/scf0920/article/details/43194947