状态压缩 poj 3254
n * m 个玉米
n*m个数字 0 或者1
1可以种玉米 0 不能 种玉米不能相邻
计算有几种 种的方法
#include<stdio.h> #include<algorithm> #include<string.h> using namespace std; #define MAXN 13 #define MAXN1 10000 #define mod 100000000 int n,m; int z[MAXN][MAXN]; int num[MAXN1]; int cnt; int dp[MAXN][MAXN1]; //dp[i][j] 第一维是行 第二维是列 表示这一行这个状态的数目 int x[MAXN]; //用来记录每一行的二进制数 1 void solve() // 0&1 =0 就1&1=1 { for(int i=0;i<(1<<m);i++) //所有可能的状态 就是这一行不可能相邻 10010 100100 这样可以 11010 110100 不行 if((i&(i<<1))==0) num[cnt++]=i; } bool jug(int state,int r) //这个状态 和这一行是否矛盾 { if(!(state&(~x[r]))) return 1; return 0; } int main() { while(scanf("%d%d",&n,&m)!=EOF) { for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) scanf("%d",&z[i][j]); cnt=0; solve(); memset(dp,0,sizeof(dp)); memset(x,0,sizeof(x)); for(int i=1;i<=n;i++) { for(int j=1;j<=m;j++) if(z[i][j]) x[i]+=(1<<(m-j)); } dp[0][0]=1; for(int i=1;i<=n;i++) { for(int st=0;st<cnt;st++) { if(jug(num[st],i)) //这一行 和 这个状态 { for(int pa=0;pa<cnt;pa++) { if(jug(num[pa],i-1)&&!(num[st]&num[pa]))//前一行和前一个状态 这和状态和前一个状态 dp[i][num[st]]+=dp[i-1][num[pa]]; } } } } int ans=0; for(int i=0;i<cnt;i++) //所有的数目要加一下 ans=(ans+dp[n][num[i]])%mod; printf("%d\n",ans); } return 0; }
posted on 2016-12-02 20:59 HelloWorld!--By-MJY 阅读(139) 评论(0) 编辑 收藏 举报