poj 2411 Mondriaan's Dream 【dp】

简介:

题目:poj 2411 Mondriaan's Dream


题意:给出一个n*m的矩阵,让你用1*2的矩阵铺满,然后问你最多由多少种不同的方案。


分析:这是一个比較经典的题目。网上各种牛B写法一大堆。题解也是

我们能够定义状态:dp【i】【st】:在第 i 行状态为 st 的时候的最慷慨案数、

然后转移方程:dp【i】【st】 = sum (dp【i-1】【ss】)

即全部的当前行都是由上一行合法的状态转移而来。

而状态的合法性由两种铺法得到。第一种横放。注意要求前一行全满。然后竖放,上一行为空。能够留空。


AC代码:

#include <cstdio>
#include <cstring>
#include <string>
#include <iostream>
#include <algorithm>
#include <vector>
using namespace std;
const long long N = 15;
long long dp[N][1<<N];
long long ans[N][N];
long long n,m;
bool solve(long long st)
{
    long long tmp=0;
    for(long long i=0; i<m; i++)
    {
        if(st&(1<<i))
            tmp++;
        else
        {
            if(tmp%2)
                return false;
            tmp=0;
        }
    }
    if(tmp%2)
        return false;
    return true;
}
bool cmp(long long st,long long up)
{
    for(long long i=0; i<m; i++)
    {
        if(st&(1<<i))
        {
            if(up&(1<<i))
            {
                if(i==m-1 || !(st&(1<<(i+1))) || !(up&(1<<(i+1))))
                    return false;
                else
                    i++;
            }//否则的话竖放
        }
        else
        {
            if(!(up&(1<<i)))
                return false;
        }
    }
    return true;
}
int main()
{
    memset(ans,0,sizeof(ans));
    while(~scanf("%lld%lld",&n,&m))
    {
        if(n==0 && m==0)
            break;
        if((m*n)%2)
        {
            printf("0\n");
            continue;
        }
        if(m>n)
            swap(m,n);
        if(ans[n][m])
        {
            printf("%lld\n",ans[n][m]);
            continue;
        }
        memset(dp,0,sizeof(dp));
        long long len = 1<<m;
        for(long long i=0; i<len; i++)
        {
            if(solve(i))
                dp[1][i]=1;
        }
        for(long long i=2; i<=n; i++)
        {
            for(long long st=0; st<len; st++) //当前行
            {
                for(long long k = 0; k<len; k++) //上一行
                {
                    if(cmp(st,k))
                        dp[i][st]+=dp[i-1][k];
                }
            }
        }
        printf("%lld\n",dp[n][len-1]);
        ans[n][m] = dp[n][len-1];
    }
    return 0;
}




本文转自mfrbuaa博客园博客,原文链接http://www.cnblogs.com/mfrbuaa/p/5226655.html,如需转载请自行联系原作者
相关文章
|
8月前
poj 3624 Charm Bracelet(简单01背包)
Given that weight limit as a constraint and a list of the charms with their weights and desirability rating, deduce the maximum possible sum of ratings.
24 0
|
10月前
|
网络架构
POJ 3250 Bad Hair Day、POJ 2796 Feel Good(单调栈)
POJ 3250 Bad Hair Day、POJ 2796 Feel Good(单调栈)
codeforces118——D. Caesar‘s Legions(DP)
codeforces118——D. Caesar‘s Legions(DP)
69 0
codeforces118——D. Caesar‘s Legions(DP)
洛谷P3009-[USACO11JAN]Profits S(DP-最大子段和)
洛谷P3009-[USACO11JAN]Profits S(DP-最大子段和)
洛谷P3009-[USACO11JAN]Profits S(DP-最大子段和)
洛谷P1216-[USACO1.5][IOI1994]数字三角形 Number Triangles(DP)
洛谷P1216-[USACO1.5][IOI1994]数字三角形 Number Triangles(DP)
洛谷P1216-[USACO1.5][IOI1994]数字三角形 Number Triangles(DP)
HDOJ(HDU) 2401 Baskets of Gold Coins(数列、)
HDOJ(HDU) 2401 Baskets of Gold Coins(数列、)
70 0
|
Java C语言
HDOJ/HDU 1029 Ignatius and the Princess IV(简单DP,排序)
HDOJ/HDU 1029 Ignatius and the Princess IV(简单DP,排序)
123 0
POJ 3624 Charm Bracelet(01背包裸题)
Charm Bracelet Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 38909   Accepted: 16862 Description Bessie has g...
1166 0
dp + 状态压缩 - Codeforces 580D Kefa and Dishes
Kefa and Dishes  Problem's Link   Mean:  菜单上有n道菜,需要点m道。每道菜的美味值为ai。 有k个规则,每个规则:在吃完第xi道菜后接着吃yi可以多获得vi的美味值。
1119 0