0x56 动态规划-状态压缩DP

Mondriaan's Dream

Description

Squares and rectangles fascinated the famous Dutch painter Piet Mondriaan. One night, after producing the drawings in his ‘toilet series’ (where he had to use his toilet paper to draw on, for all of his paper was filled with squares and rectangles), he dreamt of filling a large rectangle with small rectangles of width 2 and height 1 in varying ways.

Expert as he was in this material, he saw at a glance that he’ll need a computer to calculate the number of ways to fill the large rectangle whose dimensions were integer values, as well. Help him, so that his dream won’t turn into a nightmare!

Input

The input contains several test cases. Each test case is made up of two integer numbers: the height h and the width w of the large rectangle. Input is terminated by h=w=0. Otherwise, 1<=h,w<=11.

Output

For each test case, output the number of different ways the given rectangle can be filled with small rectangles of size 2 times 1. Assume the given large rectangle is oriented, i.e. count symmetrical tilings multiple times.

Sample Input

1 2
1 3
1 4
2 2
2 3
2 4
2 11
4 11
0 0
Null

Sample Output

1
0
1
2
3
5
144
51205
Null

题意

给出一个 n×m 的方格,问用 1×2 的小方格来填充总共有多少种方法。

思路

我们定义 dp[i][j] 代表第 i-1 行已经放满,第 i 行状态为 j 时候的方案数。

其中每一行的状态我们可以用一个二进制来表示, 0 代表未填充, 1 代表已填充。

因为方块有两种摆放形式:竖放、横放

所以当第 i 行第 j 列竖放一个方块时,第 i-1 行第 j 列需要留空;而当第 i 行第 j 列与第 j+1 列横放一个方块时,第 i-1 行第 j 列与第 j+1 列则需已填充,因为我们定义的 dp 需要把第 i-1 行全部放满。

状态转移方程: \(dp[i][s]=sum(dp[i−1][si])dp[i][s]=sum(dp[i−1][si])\) 其中状态 s 与状态 si 必须兼容,也就是状态 s 中竖放的块能够填满状态 si 中的空。

这里有一个优化,也就是当 \(n×m\)结果为奇数的时候,无论怎样都不可能成功放置,因为每一个块的面积是偶数。

// https://www.dreamwings.cn/poj2411/4615.html 千千dalao的解法
#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
using namespace std; #define MAX ((1<<11)+10)
typedef __int64 LL;
int n,m; LL dp[15][MAX];
LL ans[15][15]; bool jud(int x) // 判断 x 二进制中是否存在独立的1
{
bool is1=false;
for(int i=0; i<m; i++)
{
if(x&(1<<i))
is1=!is1;
else if(is1)return false;
}
return true;
} bool ok(int s, int ss) //判断状态s与状态ss是否兼容
{
for(int j=0; j<m; )
if(s & (1<<j)) //第i行第j列为1
{
if( ss & (1<<j)) //第i-1行第j列也为1,那么第i行必然是横放
{
//第i行和第i-1行的第j+1都必须是1,否则是非法的
if( j==m-1 || !(s&1<<(j+1)) || !(ss&(1<<(j+1))) ) return false;
else j+=2;
}
else j++; //第i-1行第j列为0,说明第i行第j列是竖放
}
else //第i行第j列为0,那么第i-1行的第j列应该是已经填充了的
{
if(ss&(1<<j)) j++;//已经填充
else return false;
}
return true;
} void solve()
{
int maxs;
if(n<m)swap(n,m); // 交换之后可以得到更小的状态数
maxs=(1<<m)-1; // 状态全1的情况
memset(dp,0,sizeof(dp));
for(int i=0; i<=maxs; i++) // 初始化第一行
dp[1][i]=jud(i);
for(int c=2; c<=n; c++) // 枚举 [2,n] 行
for(int i=0; i<=maxs; i++) // 第i行的状态
for(int si=0; si<=maxs; si++) //第i-1行的状态
if(ok(i,si))
dp[c][i]+=dp[c-1][si];
ans[n][m]=ans[m][n]=dp[n][maxs];
printf("%I64d\n",dp[n][maxs]);
} int main()
{
memset(ans,0,sizeof(ans));
while(cin>>n>>m&&(n||m))
{
if(ans[n][m]) //如果之前计算过,则直接给出结果
{
printf("%I64d\n",ans[n][m]);
continue;
}
if(n&1&&m&1) // 如果两边长都为奇数,则其面积也是奇数,无法放置
{
printf("0\n");
continue;
}
solve();
}
return 0;
}
//学习算法竞赛提升指南的写法
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
ll f[12][1 << 11];
int in_s[1 << 11];
int main() {
//freopen("in.txt", "r", stdin);
int n, m;
while (cin >> n >> m && n) {
for (int i = 0; i < 1 << m; ++i) {
bool cnt = 0, has_odd = 0;
for (int j = 0; j < m; ++j)
if (i >> j & 1) has_odd |= cnt, cnt = 0;
else cnt ^= 1;
in_s[i] = has_odd | cnt ? 0 : 1;
}
f[0][0] = 1;
for (int i = 1; i <= n; ++i)
for (int j = 0; j < 1 << m; ++j) {
f[i][j] = 0;
for (int k = 0; k < 1 << m; ++k)
if ((j & k) == 0 && in_s[j | k])
f[i][j] += f[i - 1][k];
}
cout << f[n][0] << endl;
}
}

在学习别人题解的时候发现一个DFS解决的,记录一下方法.

本质上和上面这个类似,但效率很高

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 11;
const int M = 10 + 20; int n, m, r;
ll dp[2][1 << N]; void dfs(int k, int u, int v) {
while (k < m && u & 1 << k) k++;
if (k >= m) {
dp[1 - r][v] += dp[r][u];
return;
}
dfs(k + 1, u, v | (1 << k));
if (k + 1 < m && !(u & (1 << k + 1))) {
dfs(k + 2, u, v);
}
} int main() {
ios::sync_with_stdio(false); cin.tie(0); cout.tie(0); while (cin >> n >> m && n && m) {
memset(dp, 0, sizeof(dp)); dp[1][0] = 1;
for (int i = 1; i <= n; i++) {
r = i & 1;
for (int s = 0; s < (1 << m); s++) {
if (dp[r][s]) dfs(0, s, 0);
}
memset(dp[r], 0, sizeof(dp[r]));
}
cout << dp[(n + 1) & 1][0] << endl;
} return 0;
}

炮兵阵地

思路:

待补

#include <iostream>
#include <cstring>
using namespace std;
#define MST(a, b) memset(a, b, sizeof(a));
#define CLR(a) MST(a, 0);
#define rep(x, y, z) for (int x = y; x < z; ++x)
const int INF = 0x3f3f3f3f;
int dp[101][77][77];
int sg[101];
int n, m, idx;
int s[77]; //合法摆放的集合
int cnt0[77]; //合法摆放方案的具体摆放个数, 即二进制下1的个数
int get_one(int x) {
int cnt = 0;
while (x) x &= (x - 1), ++cnt;
return cnt;
}
bool ok(int x) {
// 相邻两个P之间要有两个H
if (x & (x << 1)) return false;
if (x & (x << 2)) return false;
return true;
}
void init() {
idx = 0;
int end = 1 << m;
rep(i, 0, end) if (ok(i)) {
// s保存合法方案的集合
s[idx] = i;
// cnt0保存合法方案的摆放个数, 二进制位1的个数
cnt0[idx++] = get_one(i);
}
}
bool valid(int i, int x) {
if (sg[i] & x) return false;
return true;
}
int solve() {
int ans = 0;
MST(dp, -1);
dp[0][0][0] = 0;
rep(j, 0, idx) if (valid(1, s[j])) {
dp[1][j][0] = cnt0[j];
// 考虑n==1情况
ans = max(ans, dp[1][j][0]);
}
rep(i, 2, n + 1) {
// valid()函数判断, 第i行, 用方案s[j]是否合法
rep(j, 0, idx) if (valid(i, s[j])) {
// i行跟i-1行的方案, 满足, 互相炸不到对方
rep(k, 0, idx) if (valid(i - 1, s[k]) && (s[j] & s[k]) == 0) {
int last = 0;
// i-2行同上
rep(l, 0, idx) if (dp[i - 1][k][l] != -1 && (s[l] & s[j]) == 0 && valid(i - 2, s[l])) {
last = max(last, dp[i - 1][k][l]);
}
dp[i][j][k] = max(dp[i][j][k], last + cnt0[j]);
if (i == n) ans = max(ans, dp[i][j][k]);
}
}
}
return ans;
}
int main(int argc, char const* argv[]) {
ios::sync_with_stdio(0); cin.tie(0);
cin >> n >> m;
rep(i, 1, n + 1) rep(j, 0, m) {
char tmp; cin >> tmp;
if (tmp == 'H') sg[i] |= (1 << (m - 1 - j));
}
init();
cout << solve() << endl;
return 0;
}

# 0x56 动态规划-状态压缩DP的更多相关文章

  1. [动态规划]状态压缩DP小结

     1.小技巧 枚举集合S的子集:for(int i = S; i > 0; i=(i-1)&S) 枚举包含S的集合:for(int i = S; i < (1<<n); ...

  2. 3.4 熟练掌握动态规划——状态压缩DP

    从旅行商问题说起—— 给定一个图,n个节点(n<=15),求从a节点出发,经历每个节点仅一次,最后回到a,需要的最短时间. 分析: 设定状态S代表当前已经走过的城市的集合,显然,S<=(1 ...

  3. [知识点]状态压缩DP

    // 此博文为迁移而来,写于2015年7月15日,不代表本人现在的观点与看法.原始地址:http://blog.sina.com.cn/s/blog_6022c4720102w6jf.html 1.前 ...

  4. Vijos 1002 过河 状态压缩DP

    描述 在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青蛙可能到达的点看成数轴上 ...

  5. 状态压缩·一(状态压缩DP)

    描述 小Hi和小Ho在兑换到了喜欢的奖品之后,便继续起了他们的美国之行,思来想去,他们决定乘坐火车前往下一座城市——那座城市即将举行美食节! 但是不幸的是,小Hi和小Ho并没有能够买到很好的火车票—— ...

  6. [转]状态压缩dp(状压dp)

    状态压缩动态规划(简称状压dp)是另一类非常典型的动态规划,通常使用在NP问题的小规模求解中,虽然是指数级别的复杂度,但速度比搜索快,其思想非常值得借鉴. 为了更好的理解状压dp,首先介绍位运算相关的 ...

  7. 旅行商问题——状态压缩DP

    问题简介 有n个城市,每个城市间均有道路,一个推销员要从某个城市出发,到其余的n-1个城市一次且仅且一次,然后回到再回到出发点.问销售员应如何经过这些城市是他所走的路线最短? 用图论的语言描述就是:给 ...

  8. 状态压缩dp初学__$Corn Fields$

    明天计划上是要刷状压,但是作为现在还不会状压的\(ruoruo\)来说是一件非常苦逼的事情,所以提前学了一下状压\(dp\). 鸣谢\(hmq\ juju\)的友情帮助 状态压缩动态规划 本博文的大体 ...

  9. 浅谈状态压缩DP

    浅谈状态压缩DP 本篇随笔简单讲解一下信息学奥林匹克竞赛中的状态压缩动态规划相关知识点.在算法竞赛中,状压\(DP\)是非常常见的动规类型.不仅如此,不仅是状压\(DP\),状压还是很多其他题目的处理 ...

  10. 【算法】状态压缩DP

    状态压缩DP是什么? 答:利用位运算(位运算比加减乘除都快!)来记录状态,并实现动态规划. 适用于什么问题? 答:数据规模较小:不能使用简单的算法解决. 例题: 题目描述 糖果店的老板一共有M 种口味 ...

随机推荐

  1. Windows之——pid为4的system进程占用80端口的解决办法

    因为Apache无法启动的原因,用netstat命令查看了一下80端口是否被占用了,如下 C:\Users\Maple>netstat -ano | findstr 0.0.0.0:80 TCP ...

  2. 平台工程时代的 Kubernetes 揭秘:2023年生产状况报告深度剖析

    Kubernetes 在生产环境中的复杂性已经成为常态,在2023年这个平台工程盛行的时代,容器管理的最大亮点可能在于其灵活性,然而在运维政策和治理等方面仍然存在诸多挑战.八年过去了,在生产环境中使用 ...

  3. 2. Shell 条件测试

    重点: 条件测试. read. Shell 环境配置. case. for. find. xargs. gzip,bzip2,xz. tar. sed. 1)位置 变量 位置变量:在 bash She ...

  4. Linux速查备忘手册

    速查手册 网盘文档PDF资料: 链接: https://pan.baidu.com/s/111rqKfPaAiOHSHDo1SnckA    提取码: mhkv  1. 2.  3.  4.  5. ...

  5. 洛谷2151 [SDOI2009]HH去散步(矩阵快速幂,边点互换)

    题意:HH有个一成不变的习惯,喜欢饭后百步走.所谓百步走,就是散步,就是在一定的时间 内,走过一定的距离. 但是同时HH又是个喜欢变化的人,所以他不会立刻沿着刚刚走来的路走回. 又因为HH是个喜欢变化 ...

  6. 吉特日化MES & 实施Windows Server 远程登录的问题

    Windows远程登录提醒:由于没有远程桌面授权服务器可以提供许可证,远程会话连接已断开.请跟服务器管理员联系. 由于没有远程桌面授权服务器可以提供许可证,远程会话连接已断开.请跟服务器管理员联系. ...

  7. erp——绩效考核系统——软件需求规格说明书

    绩效考核系统--软件需求规格说明书 引言 1.1编写目的:此文件需求说明书主要是为了开发人员能了解系统之间的关系,使用者能明白系统的使用方法,另外,可以供一些学习的小白进行参考,提供需要的人参考软件需 ...

  8. 持续集成Jenkins

    一.简单慨念 持续集成(Continuous integration,简称 CI),随着近几年的发展,持续集成在项目中 得到了广泛的推广和应用. 软件集成就是用一种较好的方式,使多种软件的功能集成到一 ...

  9. Unity3d_Rewired官方文档翻译:概念(一):InputManager、Players、Actions

    仅翻译了官方文档中的Essentials(要点).Concepts(概念)两部分,这是文档中最重要的部分,理解了这两部分的内容应该足以让你将Rewired运用到你的项目中,之后再去阅读文档的其他部分也 ...

  10. 2种GaussDB(DWS)查看作业运行信息方式

    摘要:提供以作业基本单位的作业统计视图pgxc_session_wlmstat,便于用户观察运行作业和排队作业信息. 本文分享自华为云社区<GaussDB(DWS)如何查看作业运行信息>, ...