POJ 2411: Mondriaan's Dream

POJ 2411: Mondriaan's Dream

题目链接:http://poj.org/problem?id=2411

题目大意:在$h \times w$($1 \leqslant h,w \leqslant 11$)的网格中放满$1 \times 2$的木条,问有几种放置方案。

状压dp

定义横向放置的木条为$11$(横向),纵向放置的木条为$_1^0$(纵向).

这样检查转移的合法性只需要检查前一行状态$s_{i-1}$与当前行状态$s_i$的位或($s_{i-1}|s_i$)是否全为$1$,且横向的$1$是否均匹配.

定义状态$dp[i][s]$表示第$i$行状态为$s$的方案数,若状态$s_{i-1}$由状态$s_i$转移合法,则有$dp[i][s_{i-1}]+=dp[i-1][s_i]$.

最后一行不能放置向下的纵向木条,即最后一行全为$1$,故$ans=dp[n-1][(1<<m)-1]$.

可以预处理合法的状态并标记以降低复杂度。

总时间复杂度为$O(2^{MAX\{m\}}+n2^{2m})$.

代码如下:

 1 #include <cstdio>
 2 #include <cstring>
 3 using namespace std;
 4 typedef long long ll;
 5 int n,m;
 6 ll dp[15][1<<11];
 7 bool s[1<<11];
 8 bool valid(int s){
 9     int tot=0;
10     while(s){
11         if(s&1)tot++;
12         else if(tot&1)return 0;
13         s>>=1;
14     }
15     return tot%2==0;
16 }
17 void init(){
18     for(int i=0;i<(1<<11);++i)
19         s[i]=valid(i);
20 }
21 bool check(int x,int y){
22     if((x|y)+1!=(1<<m))return 0;
23     return s[x&y];
24 }
25 int main(void){
26     init();
27     while(~scanf("%d%d",&n,&m)){
28         if(n==0&&m==0)break;
29         memset(dp,0,sizeof(dp));
30         for(int i=0;i<(1<<m);++i)
31             if(s[i])dp[0][i]=1;
32         for(int i=1;i<n;++i){
33             for(int j=0;j<(1<<m);++j)if(dp[i-1][j]){
34                 for(int k=0;k<(1<<m);++k)if(check(j,k)){
35                     dp[i][k]+=dp[i-1][j];
36                 }
37             }
38         }
39         printf("%lld\n",dp[n-1][(1<<m)-1]);
40     }
41 }

 

posted @ 2017-04-26 17:23  barriery  阅读(234)  评论(0编辑  收藏  举报