poj3254 状态压缩dp

题意:给出一个n行m列的草地,1表示肥沃,0表示贫瘠,现在要把一些牛放在肥沃的草地上,但是要求所有牛不能相邻,问你有多少种放法。
 

 
分析:假如我们知道第 i-1 行的所有的可以放的情况,那么对于第 i 行的可以放的一种情况,我们只要判断它和 i - 1 行的所有情况的能不能满足题目的所有牛不相邻,如果有种中满足,那么对于 i 行的这一中情况有 x 中放法。 但是我们又发现,状态是一种放法,不是我们平常dp的简单的状态,所以要用状态压缩!
 
dp[i][j]表示第i行状态为j的个数。
dp[i][j] = sum{dp[i-1][k]};
第i行可以由上一行的合法的状态推导来。
 
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define INF 1000000007
#define mod 100000000
#define ll long long
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
using namespace std;
const int MAXN = 13;
int n,m;
int mp[MAXN];
int dp[MAXN][(1<<MAXN)];
int q[1<<MAXN],cnt;
int main()
{
    while(~scanf("%d%d",&n,&m)){
        int x;
        cnt = 0;
        memset(mp,0,sizeof(mp));
        for(int i = 1; i <= n; i++){
            for(int j = 1; j <= m; j++){
                scanf("%d",&x);
                if(x == 0){
                    mp[i] |= (1<<(j-1));//mp[i]存合法的状态 x=1的时候,用0存,0的时候用1存。
                }
            }
        }
        memset(dp,0,sizeof(dp));//dp表示第i行状态为j时合法的个数
        for(int i = 0; i < (1 << m); i++){//在一行中把没有相邻的合法的状态记录下来
            if(!(i & (i << 1))){
                q[cnt++] = i;
            }
        }
        for(int i = 0; i < cnt; i++){//对第一行初始化。由于mp存第i行合法的状态,又0用1 1用0表示,所以mp[1]&q[i]==0
                                     //表示和合法的状态
            if(!(mp[1] & q[i])){
                dp[1][i] = 1;
            }
        }
        for(int i = 2; i <= n; i++){//从第2行开始枚举
            for(int j = 0; j < cnt; j++){//枚举状态
                if(mp[i] & q[j]){//不合法的状态
                    continue;
                }
                for(int k = 0; k < cnt; k++){
                    if((mp[i-1] & q[k]) == 0 && (q[j] & q [k]) == 0){
                        dp[i][j] = (dp[i][j] + dp[i-1][k])%mod;
                    }
                }
            }
        }
        ll ans = 0;
        for(int i = 0; i < cnt; i++){
            ans =(ans + dp[n][i]) % mod;
        }
        cout<<ans<<endl;
    }
    return 0;
}
原文地址:https://www.cnblogs.com/sweat123/p/5413989.html