BZOJ 1725: [Usaco2006 Nov]Corn Fields牧场的安排 状压动归

Description

Farmer John新买了一块长方形的牧场,这块牧场被划分成M列N行(1<=M<=12; 1<=N<=12),每一格都是一块正方形的土地。FJ打算在牧场上的某几格土地里种上美味的草,供他的奶牛们享用。遗憾的是,有些土地相当的贫瘠,不能用来放牧。并且,奶牛们喜欢独占一块草地的感觉,于是FJ不会选择两块相邻的土地,也就是说,没有哪两块草地有公共边。当然,FJ还没有决定在哪些土地上种草。 作为一个好奇的农场主,FJ想知道,如果不考虑草地的总块数,那么,一共有多少种种植方案可供他选择。当然,把新的牧场荒废,不在任何土地上种草,也算一种方案。请你帮FJ算一下这个总方案数。

Input

* 第1行: 两个正整数M和N,用空格隔开

* 第2..M+1行: 每行包含N个用空格隔开的整数,描述了每块土地的状态。输入的第i+1行描述了第i行的土地。所有整数均为0或1,是1的话,表示这块土地足够肥沃,0则表示这块地上不适合种草

Output

* 第1行: 输出一个整数,即牧场分配总方案数除以100,000,000的余数

 

题解: 

状压动归傻题.

保存有用状态来减少枚举的状态数.

Code:

#include<bits/stdc++.h>
#define setIO(s) freopen(s".in","r",stdin) 
#define mod 100000000
#define maxn 20 
#define ll long long 
using namespace std;      
ll F[15][20000],ans; 
int G[20][20],w[20]; 
vector<int>sta[maxn],h; 
int main(){
    // setIO("input"); 
    int n,m; 
    scanf("%d%d",&n,&m); 
    for(int i=1;i<15;++i) w[i]=(1<<(i-1)); 
    //预处理出所有合法状态 
    for(int k=0;k<w[m+1];++k)
    {
        int l=k,pre=0,flag=1; 
        while(l)
        {
            if((l&1) && pre) { flag=0; break; }
            if(l&1) 
                pre = 1;  
            else 
                pre = 0; 
            l>>=1; 
        }
        if(flag) h.push_back(k); 
    }
    for(int i=1;i<=n;++i)
    {
        int cur=0; 
        for(int j=1;j<=m;++j) 
        {
            scanf("%d",&G[i][j]);               
            if(G[i][j]==0) cur|=w[j];    //不合法
        }        
        for(int j=0,sz=h.size();j<sz;++j) 
        {
            if(h[j]&cur) continue; 
            sta[i].push_back(h[j]); 
        }
    }
    F[1][0]=1; 
    for(int i=0,sz=sta[1].size();i<sz;++i) F[1][sta[1][i]]=1; 
    for(int i=2;i<=n;++i)
    {
        for(int j=0,sz=sta[i].size();j<sz;++j) 
        {                        
            for(int k=0,sz2=sta[i-1].size();k<sz2;++k){
                if(sta[i][j]&sta[i-1][k]) continue; 
                F[i][sta[i][j]]=(F[i][sta[i][j]]+F[i-1][sta[i-1][k]])%mod; 
            } 
        }
    }         
    for(int i=0,sz=sta[n].size();i<sz;++i) ans=(ans+F[n][sta[n][i]])%mod;             
    printf("%lld",ans); 
    return 0; 
}

  

posted @ 2019-05-30 10:47  EM-LGH  阅读(166)  评论(0编辑  收藏  举报