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


解题心得:

  1. 可以说是很经典的一个状压dp了,写dfs遍历肯定是要超时的,这个题的状态转移方程对新手来说有点吃力。
  2. 状态转移用的是上凸法,就是如果一行中(除了最后一行)有一个是空位,那么必定是下面一行竖着摆放的矩形,这样才符合条件。这样就把两行中的状态联系起来了,枚举每两行的状态来进行检验,还可以进行状态的累加。
  3. 这个题还有一些小技巧,
    • 如果列数大于行数可以将列和行互换,因为在计算行和列的复杂度是完全不同的,列是o(2^n),行是o(n)。,
    • 另一个就是因为行列数量级很小,测试样例可能很多,可以把每次的答案记录下来,这样在有答案的时候可以直接输出。
    • -


#include<stdio.h>
#include<cstring>
#include<iostream>
using namespace std;
const int maxn = (1<<11)+10;
long long dp[15][maxn],ans[15][15];
int n,m,Max; bool init(int x)
{
int i=0;
while(i<m)
{
if(x&(1<<i))
{
if(i == m-1)//不能超出边界
return false;
if(x&(1<<(i+1)))//一个横放的矩形
i += 2;
else
return false;
}
else
i++;
}
return true;
} bool ok(int x,int y)
{
int i = 0;
while(i<m)
{
if(x&(1<<i))//两个都是横放的
{
if(y&(1<<i))
{
if(i == m-1 || !(x&(1<<(i+1))) || !(y&(1<<(i+1))))
return false;
else
i+=2;
}
else
i++;
}
else//当前行有放置的矩形,上一行是空
{
if(y&(1<<i))
i++;
else
return false;
}
}
return true;
} void solve()
{
Max = (1<<m) -1;
memset(dp,0,sizeof(dp));
for(int i=0;i<=Max;i++)//特殊的第一行初始化
{
if(init(i))
dp[0][i] = 1;
}
for(int i=1;i<n;i++)
for(int s=0;s<=Max;s++)//枚举每两行符合条件的放置方式
for(int ss=0;ss<=Max;ss++)
{
if(ok(s,ss))
dp[i][s] += dp[i-1][ss];
}
printf("%lld\n",ans[n][m] = ans[m][n] = dp[n-1][Max]);
} int main()
{
memset(ans,0,sizeof(ans));
while(cin>>n>>m)
{
if(m > n)//如果行小于列,直接交换行和列
swap(n,m);
if(n+m == 0)
break;
if(ans[n][m] || ans[m][n])//记录已经出现过的答案
{
printf("%lld\n",ans[n][m]);
continue;
}
if((n*m)%2 == 1)//如果行列都是奇数那么没有符合条件的方案
{
printf("0\n");
continue;
}
solve();
}
}

POJ:2411-Mondriaan's Dream(矩形拼接方案)的更多相关文章

  1. POJ 2411 Mondriaan's Dream 插头dp

    题目链接: http://poj.org/problem?id=2411 Mondriaan's Dream Time Limit: 3000MSMemory Limit: 65536K 问题描述 S ...

  2. Poj 2411 Mondriaan's Dream(压缩矩阵DP)

    一.Description Squares and rectangles fascinated the famous Dutch painter Piet Mondriaan. One night, ...

  3. POJ 2411 Mondriaan's Dream -- 状压DP

    题目:Mondriaan's Dream 链接:http://poj.org/problem?id=2411 题意:用 1*2 的瓷砖去填 n*m 的地板,问有多少种填法. 思路: 很久很久以前便做过 ...

  4. POJ - 2411 Mondriaan's Dream(轮廓线dp)

    Mondriaan's Dream Squares and rectangles fascinated the famous Dutch painter Piet Mondriaan. One nig ...

  5. [POJ] 2411 Mondriaan's Dream

    Mondriaan's Dream Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 18903 Accepted: 10779 D ...

  6. Poj 2411 Mondriaan's Dream(状压DP)

    Mondriaan's Dream Time Limit: 3000MS Memory Limit: 65536K Description Squares and rectangles fascina ...

  7. [poj 2411]Mondriaan's Dream (状压dp)

    Mondriaan's Dream Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 18903 Accepted: 10779 D ...

  8. POJ 2411 Mondriaan's Dream/[二进制状压DP]

    题目链接[http://poj.org/problem?id=2411] 题意:给出一个h*w的矩形1<=h,w<=11.用1*2和2*1的小矩形去填满这个h*w的矩形,问有多少种方法? ...

  9. poj 2411 Mondriaan's Dream(状态压缩dp)

    Description Squares and rectangles fascinated the famous Dutch painter Piet Mondriaan. One night, af ...

  10. poj 2411 Mondriaan's Dream 轮廓线dp

    题目链接: http://poj.org/problem?id=2411 题目意思: 给一个n*m的矩形区域,将1*2和2*1的小矩形填满方格,问一共有多少种填法. 解题思路: 用轮廓线可以过. 对每 ...

随机推荐

  1. python之三级菜单

    python之三级菜单 要求: 1. 运行程序输出第一级菜单 2. 选择一级菜单某项,输出二级菜单,同理输出三级菜单 3. 菜单数据保存在文件中 4. 让用户选择是否要退出 5. 有返回上一级菜单的功 ...

  2. DevExpress GridControl 控件二表连动

    using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...

  3. AXURE-手把手教你做汉化

    ​ 我们默认下载的AXURE是英文版的,对于英文能力不足或者不习惯英文界面的,那必须使用汉化手段,网上也有很多朋友已经为大家做好了汉化文件,这里介绍一下如何自己做AXURE的汉化. 如何开始汉化 如何 ...

  4. java 使用uuid生成唯一字符串

    UUID(Universally Unique Identifier)全局唯一标识符,是指在一台机器上生成的数字,它保证对在同一时空中的所有机器都是唯一的.按照开放软件基金会(OSF)制定的标准计算, ...

  5. iOS中转义后的html标签如何还原

    最近用swift做一个公司的小项目,遇到一个问题,就是通过api获取的html文本中的标签都已经被转义了, <p class="MsoNormal" align=" ...

  6. Codeforces Round #411 div2

    A. Fake NP 题意:询问一个区间[L,R]出现次数最多的正整数因子(>1). 一个区间内一个因子P出现次数大概为[R/P]-[(L-1)/P],约等于(R-L+1)/P,P取2时最优.注 ...

  7. cesium模型加载-加载fbx格式模型

    整体思路: fbx格式→dae格式→gltf格式→cesium加载gltf格式模型 具体方法: 1. fbx格式→dae格式 工具:3dsMax, 3dsMax插件:OpenCOLLADA, 下载地址 ...

  8. UVALive 3942 Remember The Word (Tire)

    状态是DAG,因此方案用dp统计,dp[i] = sum(dp[i+len(x)]),x是以i开头的前缀且是单词,关键在于快速判断一个前缀是不是单词,可用Trie. 每一次转移的复杂度是O(maxle ...

  9. CDOJ 490 UESTC 490 Swap Game(思路,逆序对)

    题意:有两种颜色的小球形成环,求最小交互次数使球相连. 题解:先解决另一个简单的问题,如果是一个链,把红球标记为1,蓝球标记为0,要排成升序需要多少次交换呢?答案是逆序对总数,原因是一次交互最多消除一 ...

  10. Python中Numpy mat的使用

    前面介绍过用dnarray来模拟,但mat更符合矩阵,这里的mat与Matlab中的很相似.(mat与matrix等同) 基本操作 >>> m= np.mat([1,2,3]) #创 ...