铺砖问题 (状态压缩dp)
问题描述:
给定m×n个格子,每个格子被染成了黑色或白色。现在要用1×2的砖块覆盖这些格子,要求快于快之间互相不重叠,且覆盖了所有白色的格子(用 . 表示),但不覆盖任意一个黑色的格子(用 x 表示)。求一共有多少种覆盖方法,输出方案数对M取余后的结果。
首先考虑一下枚举所有的解法这一方法。为了不重复统计,我们每次从最左上方的空格处开始放置。对于哪些格子已经被覆盖过了,我们只需要用一个bool数组来维护即可,即:
#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std;
int n,m,M;
char tu[20][20];
bool used[20][20];
int rec(int i,int j,bool used[20][20])
{
if(j==m)///当前这一行已经走完了
{
return rec(i+1,0,used);
}
if(i==n)///已经把整个图都走完了
{
return 1;
}
if(used[i][j]||tu[i][j]=='x')///当这个点已经放过或者说是黑色的,则不需要放置地砖
{
return rec(i,j+1,used);
}
int res=0;
used[i][j]=true;///标记这个点已经放过了
///横着放
if(j+1<m&&!used[i][j+1]&&tu[i][j+1]=='.')///横着放能够放
{
used[i][j+1]=true;///笔记后面那个格子
res+=rec(i,j+1,used);///接着往下找
used[i][j+1]=false;///标记释放
}
///竖着放
if(i+1<n&&!used[i+1][j]&&tu[i+1][j]=='.')///竖着放能够放
{
used[i+1][j]=true;///标记下面那个格子
res+=rec(i ,j+1,used);///刚开始以为这里应该是rec(i,j+1,used),后来发现理解错了,i,j只是表示往后走
used[i+1][j]=false;///标记释放
}
used[i][j]=false;///当前格子的标记释放
return res%M;
}
void solve()
{
memset(used,0,sizeof(used));
printf("%d\n",rec(0,0,used));
}
int main()
{
while(~scanf("%d%d%d",&n,&m,&M))
{
for(int i=0; i<n; i++)
scanf(" %s",tu[i]);
solve();
}
return 0;
}
这个方法无法在规定的时间内求出答案。而且递归的参数共有n×m×2^nm种可能,也无法使用记忆话搜索。
首先,由于黑色的格子不能被覆盖,因此used里对应的 位置总是false。对于白色的格子,如果要在(i,j)位置上放置砖块,那么由于总是从最左上方可以放的位置开始放,那么对于(i',j')<(i,j)(按字典序比较)的(i',j')总有used[i'][j']=true成立。
此外,由于砖块的大小为1×2,因此对于每一列j'在满足(i',j')>=(i,j)的所有i'中,除了最小的i‘之外,都满足used[i'][j']=false。因此,不确定的只有每一列里还没有查询格子中最上面的一个,共m个。从而把这m个格子通过状态压缩编码进行记忆化搜索,复杂度为(n×m×2^m)。
#include<iostream>
#include<stdio.h>
#include<string.h>
#include<cmath>
using namespace std;
int n,m,M;
char tu[20][20];
//bool used[20][20];
int dp[2][1<<20];
void solve()
{
int *crt=dp[0],*next=dp[1];
crt[0]=1;
for(int i=n-1; i>=0; i--)
{
for(int j=m-1; j>=0; j--)
{
for(int used=0; used<1<<m; used++)
{
///不需要在[i,j]放置砖块
if((used>>j&1)||tu[i][j]=='x')
{
next[used]=crt[used&~(1<<j)];
}
else
{
int res=0;
///横着放
if(j+1<m&&!(used>>(j+1)&1)&& tu[i][j+1]=='.')
{
res+=crt[used|1<<(j+1)];
}
///竖着放
if(i+1<n&&tu[i+1][j]=='.')
{
res+=crt[used|1<<j];
}
next[used]=res%M;
}
}
swap(crt,next);
}
}
printf("%d\n",crt[0]);
}
int main()
{
while(~scanf("%d%d%d",&n,&m,&M))
{
for(int i=0; i<n; i++)
scanf(" %s",tu[i]);
solve();
}
return 0;
}
铺砖问题 (状态压缩dp)的更多相关文章
- ZOJ 2563 Long Dominoes(状态压缩DP)
题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1563 题目大意:在h*w的矩阵里铺满1*3的小矩阵,共有多少种方法 ...
- SGU131 - Hardwood floor(状态压缩DP)
题目大意 给定一个N*M大小的矩形,要求你用1*2和2*2(缺个角)的砖块把矩形铺满(不能重叠),问总共有多少种铺法? 题解 受POJ2411的影响,怎么都没想到3,4,5,6这几种情况该怎么放置,看 ...
- hoj2662 状态压缩dp
Pieces Assignment My Tags (Edit) Source : zhouguyue Time limit : 1 sec Memory limit : 64 M S ...
- POJ 3254 Corn Fields(状态压缩DP)
Corn Fields Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 4739 Accepted: 2506 Descr ...
- [知识点]状态压缩DP
// 此博文为迁移而来,写于2015年7月15日,不代表本人现在的观点与看法.原始地址:http://blog.sina.com.cn/s/blog_6022c4720102w6jf.html 1.前 ...
- HDU-4529 郑厂长系列故事——N骑士问题 状态压缩DP
题意:给定一个合法的八皇后棋盘,现在给定1-10个骑士,问这些骑士不能够相互攻击的拜访方式有多少种. 分析:一开始想着搜索写,发现该题和八皇后不同,八皇后每一行只能够摆放一个棋子,因此搜索收敛的很快, ...
- DP大作战—状态压缩dp
题目描述 阿姆斯特朗回旋加速式阿姆斯特朗炮是一种非常厉害的武器,这种武器可以毁灭自身同行同列两个单位范围内的所有其他单位(其实就是十字型),听起来比红警里面的法国巨炮可是厉害多了.现在,零崎要在地图上 ...
- 状态压缩dp问题
问题:Ignatius has just come back school from the 30th ACM/ICPC. Now he has a lot of homework to do. Ev ...
- BZOJ-1226 学校食堂Dining 状态压缩DP
1226: [SDOI2009]学校食堂Dining Time Limit: 10 Sec Memory Limit: 259 MB Submit: 588 Solved: 360 [Submit][ ...
- Marriage Ceremonies(状态压缩dp)
Marriage Ceremonies Time Limit:2000MS Memory Limit:32768KB 64bit IO Format:%lld & %llu ...
随机推荐
- java List接口实现类
首先看这两类都实现List接口,而List接口一共有三个实现类,分别是ArrayList.Vector和LinkedList.List用于存放多个元素,能够维护元素的次序,并且允许元素的重复.3个具体 ...
- intelliJ IDEA最常用的快捷键
一.使用相关快捷键 1.重写接口实现类:Ctrl+I 2.搜索:Shift+Shift 3.生成get或set方法快捷键:Alt+insert: 4.导入未实现的方法,强制类型转换:Alt+Ent ...
- Xcode开发技巧之code snippets(代码片段)
一.什么是代码片段 当在Xcode中输入dowhile并回车后,Xcode会出现下图所示的提示代码: 这就是代码片段,目的是使程序员以最快的速度输入常用的代码片段,提高编程效率.该功能是从Xcode4 ...
- 【bzoj5094】硬盘检测 乱搞
题目描述 已知从 $n$ 个不同的32位无符号整数中随机选 $m=10000$ 次所得的结果,求可能性最大的 $n$ ,其中 $n=10^k,1\le k\le 7$. 输入 第一行包含一个正整数m( ...
- 关于"作数类型冲突: nvarchar 与 image 不兼容"的问题
数据库如果是Image类型,当执行插入语句时,如果插入的值是DBNull.Value时提示:操作数类型冲突: nvarchar 与 image 不兼容; 出现这个问题的原因是没有指定DbType的原因 ...
- [CQOI2009]跳舞 网络流
题面:[CQOI2009]跳舞 题解: 首先最大时间不好求,而且数据范围很小,所以我们可以先二分一个最大时间,然后就只需要判断是否可行即可. 因此我们每二分一个mid,对于每个女生,连s ---> ...
- CF311B Cats Transport 斜率优化DP
题面:CF311B Cats Transport 题解: 首先我们观察到山与距离其实是没有什么用的,因为对于任意一只猫,我们都可以直接算出如果有一个人要恰好接走它,需要在哪一时刻出发,我们设第i只猫对 ...
- POJ1422:Air Raid——题解
http://poj.org/problem?id=1422 题目大意:n个点m条有向边,每条边只能走一次,往点上放人让他们走遍所有边,问至少要多少人. —————————————————————— ...
- LeetCode中二叉树题目总结
本文仅为博主个人总结,水平有限,欢迎大神指出不妥处. 关于二叉树的相关概念可以参见二叉树的百度百科,或binary tree Wiki. 二叉树结点类的常见定义为: /* Definition for ...
- 卡特兰数(Catalan Number) 学习笔记
一.三个简单的问题 1.给定一串长为2n的01序列,其中0和1的数量相等,满足任意前缀中0的个数不少于1的个数,求序列的个数 2.给出一串长为n的序列,按顺序将他们进栈,随意出栈,求最后进出栈的方案 ...