超经典的一道题目,实现这题的方法也有非常多种

1.利用DFS建立矩阵,然后通过高速矩阵幂得到答案(运用于min(m,n)比較小。可是max(m,n)很大的情况)

2.利用dp状压解决

第一种在我的还有一篇博客里有

http://blog.csdn.net/qwb492859377/article/details/47138821

我们在这里讨论另外一种方法。

刘汝佳的算法训练指南中384页介绍了一种复杂度O(mn*2^m)的方法,这个复杂度实在是太小了。

我对着上面敲结果仅仅用了32ms就过了这题,然后花了非常久才看懂(蒟蒻)

他的状态并非一整行一整行的。。所以一開始我就陷入了误圈。

第二,,他代码里有个1<<m,我一直没搞清楚。究竟他的下标是从0開始还是从1開始,怎么会出现1<<m呢。事实上这是上一次的1<<m-1。由于他把状态先左移过一次-_-....

我依据我的理解讲下代码

(i,j)表示以这个格子为右下角,要不要放骨牌以及放那种骨牌. k表示在(i,j)之前的连续m个方块的状态,0表示没放,1表示已经放了,当中靠近(i,j)的下标是0

先枚举,i再枚举j,最后枚举k,也就是红色表示的区域

假设(i,j)不放。那么它上面那一个就必须已经放过了,换句话说,仅仅有k&(1<<(m-1)) 。k才有资格将状态转移到新的状态(k<<1)^(1<<m)

这个状态就是左移后。把曾经的第一位去掉。然后在最后面补上一个0,相当于红色区域所有向右挪了一个单位。

假设(i,j)放竖直的骨牌,那么i不能为0,且!k&(1<<(m-1))满足。才干考虑把k状态转移到新的状态k << 1 | 1去

假设(i,j)放横向的骨牌,那么j不能为0,且!(k&1)&&(1<<(m-1)) 也就是k的最后一个要是空着的。并且(i,j)上面一个必须是1。这样才干横向放后上面的不会存在空着的。

刚開始我还在思考,应该(i-1,j-1)也应该要是1才干够把。,可是后来想了一下。那个位置一定会是1,否则(i,j)左边那个位置就不可能是0了。由于仅仅能通过第一种情况转移过来

所以当满足这些条件时,k能转移到新状态(k<<1)^(1<<m)|3上来

语言表达能力有限,,,来看代码把,,感觉刘汝佳那份代码尽管写起来非常工整清晰。反而有些地方变得非常难理解了

然后dp利用滚动数组,得到的状态向后更新答案的方式,就能攻克了

#include<cstdio>
#include<cmath>
#include<cstring>
#include<queue>
#include<vector>
#include<functional>
#include<algorithm> using namespace std;
typedef long long LL;
typedef pair<int, int> PII; const int MX = 3000 + 5;
const int INF = 0x3f3f3f3f; int m, n;
LL dp[2][MX]; int main() {
//freopen("input.txt","r",stdin);
while(~scanf("%d%d", &m, &n), m + n) {
memset(dp, 0, sizeof(dp));
if(m > n) swap(m, n); int cur = 0, nxt = 1;
dp[cur][(1 << m) - 1] = 1; for(int i = 0; i < n; i++) {
for(int j = 0; j < m; j++) {
memset(dp[nxt], 0, sizeof(dp[nxt])); for(int k = 0; k < (1 << m); k++) {
if(k & (1 << (m - 1))) dp[nxt][(k << 1) ^ (1 << m)] += dp[cur][k];//不放
if(i && !(k & (1 << (m - 1)))) dp[nxt][k << 1 | 1] += dp[cur][k];//竖着放
if(j && !(k & 1) && (1 << (m - 1))) dp[nxt][(k << 1) ^ (1 << m) | 3] += dp[cur][k];//横着放
}
swap(cur, nxt);
}
}
printf("%I64d\n", dp[cur][(1 << m) - 1]);
}
return 0;
}

状压dp Mondriaan&#39;s Dream poj2411的更多相关文章

  1. POJ 2411 Mondriaan&#39;s Dream

    状压DP Mondriaan's Dream Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 9938 Accepted: 575 ...

  2. [poj2411] Mondriaan's Dream (状压DP)

    状压DP Description Squares and rectangles fascinated the famous Dutch painter Piet Mondriaan. One nigh ...

  3. [Poj2411]Mondriaan's Dream(状压dp)(插头dp)

    Mondriaan's Dream Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 18096   Accepted: 103 ...

  4. poj2411 Mondriaan's Dream (轮廓线dp、状压dp)

    Mondriaan's Dream Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 17203   Accepted: 991 ...

  5. poj2411 Mondriaan's Dream[简单状压dp]

    $11*11$格子板上铺$1*2$地砖方案.以前做过?权当复习算了,毕竟以前学都是浅尝辄止的..常规题,注意两个条件:上一行铺竖着的则这一行同一位一定要铺上竖的,这一行单独铺横的要求枚举集合中出现连续 ...

  6. POJ2411 Mondriaan's Dream 【状压dp】

    没错,这道题又是我从LZL里的博客里剽过来的,他的题真不错,真香. 题目链接:http://poj.org/problem?id=2411 题目大意:给一个n * m的矩形, 要求用 1 * 2的小方 ...

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

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

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

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

  9. POJ 2411 Mondriaan's Dream ——状压DP 插头DP

    [题目分析] 用1*2的牌铺满n*m的格子. 刚开始用到动规想写一个n*m*2^m,写了半天才知道会有重复的情况. So Sad. 然后想到数据范围这么小,爆搜好了.于是把每一种状态对应的转移都搜了出 ...

随机推荐

  1. hdu 4071& poj 3873 & zoj 3386 & uva 12197 Trick or Treat 三分法

    思路: 看到这个题目就发现所需最短时间也就是房子和相遇点的最远距离具有凹凸性,很容易就想到了三分法枚举. 找出所有房子的X坐标的最小最大值作为上下界. 代码如下: #include<stdio. ...

  2. 微信小程序-微信自动退款(Java后台)

    微信小程序-微信自动退款 1.首先分享 微信自动退款接口: https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_4 微信付款 代码案例 ...

  3. java阶乘问题

    问题描述: 编写代码求:1!+2!+3!+…+20!的值 代码 public class Demo { public static void main(String[] args) { long nu ...

  4. 84. CYD啃骨头(背包问题)

    3111 CYD啃骨头  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 黄金 Gold 题解 查看运行结果 题目描述 Description CYD吃饭时有N个骨头可以啃,但C ...

  5. 关于JS的This指针

    下面讨论一个执行上下文的最后一个属性——this指针的概念. This指针 A this value is a special object which is related with the exe ...

  6. JavaScript基础入门教程(二)

    说明 前一篇博客介绍了js以及一些关于js基本类型的简单知识,本篇博客将详细介绍js的基础类型,捎带介绍对象类型,更详细的对象类型的说明将后续再讲. js中类型的说明 js中的类型分为基本类型和对象类 ...

  7. [UML] UML中类之间的几种关系

    类之间可能存在以下几种关系:关联(association).依赖(dependency).聚合(Aggregation,也有的称聚集).组合(Composition).泛化(generalizatio ...

  8. 各种软核处理器二进制文件FPGA初始化文件生成程序

    不管是MIPS, Nios II, MicroBlaze, MSP430, 8051, OpenRISC, OpenSPARC, LEON2/LEON3等等软核处理器,在FPGA上实现的时候我们通常需 ...

  9. xss编码小结

    一.JS编码与HTML编码区分: HTML实体可以使用十进制与十六进制编码:javascript可以使用Unicode与八进制与十六进制进行编码. 二.编码原理区分: 三.编码与非编码 对于JS编码: ...

  10. 【实践】关于p 标签内嵌 p标签的bug

    项目中遇到了一点小问题: 是这样的,在输入框包裹元素 p标签中想内嵌一个p 标签用作显示提示字符,谁知发生了一下一幕: 页面结构: <p class="modify-info-wrap ...