状压dp
第一次写状压dp的题目,可能是因为这种类型的比较难吧,所以是第一次做到。
题解:
我们可以用二进制来描述方格,1表示有德玛西亚,0表示没有。
样例1中第一行的三个方格都可以站人
一共有五种情况分别是

第一行的五种情况
第一种    000
第二种    001
第三种    010
第四种    100
第五种    101

再看第二行

第二行情况
第一种    000
第二种    010
第三种    001

第二行的第一种情况是有5种方法的,第二种情况是有4种方法(与第一行第三种情况冲突),第三种情况是有3种方法(与第一行第2、5钟情况冲突)

第三行情况

第一种    000
第二种    100

第三行的第一种情况可以完美兼容第二行所有情况,所以一共是5+4+3=12钟情况
第三行的第二种情况也是可以完美兼容第二行的所有情况(没有冲突就视为可以),所以一共是5+4+3=12种情况
综上所述,总共的情况就是24种。
所以我们先把每一行能放东西的格子用二进制表示出来。
再枚举这个长度所有的二进制状态(表示每种占位)并且将可行的占位记录下来
要判断是否与上面一行冲突和与左右两边冲突是否与地图样貌冲突

if(((j&lines[i])!=j)||(j&(j>>1))) continue;  //判断是否与地图匹配/是否左右相邻
if(k&j) continue;                            //判断是否与上层冲突

代码:

#pragma GCC optimize(3,"Ofast","inline")
#include 
const int maxn = 15;
const int MaxN = 0x3f3f3f3f;
const int MinN = 0xc0c0c00c;
typedef long long ll;
const int mod = 100000000;
using namespace std;
int dp[maxn][1<<maxn];
int lines[maxn];
int main()
{
    int n,m,x;
    while(cin>>n>>m){
        memset(lines,0,sizeof(lines));
        memset(dp,0,sizeof(dp));
        dp[0][0]=1;
        for(int i=1;i<=n;i++){
            for(int j=0;j<m;j++){
                scanf("%d",&x);
                lines[i]|=(x<<j);
            }
        }
        for(int i=1;i<=n;i++){
            for(int j=0;j<(1<<m);j++){
                if(((j&lines[i])!=j)||(j&(j>>1))) continue;
                for(int k=0;k<(1<<m);k++){
                    if(k&j) continue;
                    dp[i][j]=(dp[i][j]+dp[i-1][k])%mod;
                }
            }
        }
        ll ans=0;
        for(int i=0;i<(1<<m);i++){
            ans=(ans+dp[n][i])%mod;
        }
        cout<<ans<<endl;
    }
    return 0;
}