dp合集 广场铺砖问题&&硬木地板


很经典了吧。。。

前排:思想来自yali朱全民dalao的ppt百度文库免费下载

后排:STO朱全民OTZ


广场铺砖问题

有一个 W 行 H 列的广场,需要用 1*2 小砖铺盖,小砖之间互相不能重叠,问

有多少种不同的铺法?

输入数据:

只有一行 2 个整数,分别为 W 和 H,( 1<=W, H<=11)

输出数据:

只有 1 个整数,为所有的铺法数。

样例:

Floor.in

2 4

Floor.out

5

dfs、bfs。。。算了吧

然而我看了一眼ppt,这不是SBT吗???

然后就写出来了

设f[i][j]表示第i行,状态为j的转移方法数

具体思想:一个状态有两种方式转移:全竖放转移and横放一个转移。

然后WA了

原因:横放可能有重复(比如□□□□,可以先左边两个也可以先右边两个),然后GG

// It is made by XZZ
#include<cstdio>
#include<algorithm>
#define lb(a) (a&-a)
#define Fname "floor"
using namespace std;
#define rep(a,b,c) for(rg int a=b;a<=c;a++)
#define drep(a,b,c) for(rg int a=b;a>=c;a--)
#define erep(a,b) for(rg int a=fir[b];a;a=nxt[a])
#define il inline
#define rg register
#define vd void
typedef long long ll;
il int gi(){
rg int x=0;rg char ch=getchar();
while(ch<'0'||ch>'9')ch=getchar();
while(ch>='0'&&ch<='9')x=x*10+ch-'0',ch=getchar();
return x;
}
int f[12][1<<11];
int cnt[1<<11],s[1<<11];
il bool cmp(int a,int b){return cnt[a]<cnt[b];}
int main(){
#ifdef xzz
freopen(Fname".in","r",stdin);
freopen(Fname".out","w",stdout);
#endif
int n=gi(),m=gi();
if((n*m)&1){puts("0");return 0;}
f[0][0]=1;
int tot=(1<<m)-1;
rep(i,0,tot){
int j=i;
while(j)++cnt[i],j-=lb(j);
}
rep(i,0,tot)s[i]=i;
sort(s+1,s+tot+1,cmp);
int g,j;
rep(i,0,n-1)rep(jj,0,tot){
j=s[jj];
g=f[i][j];
printf("f[%d][%d]=%d\n",i,j,f[i][j]);
f[i+1][(~j)&tot]+=g;//所有的都竖放
rep(k,0,m-2)if(!((1<<k)&j)&&!((1<<k+1)&j))f[i][j|(1<<k)|(1<<k+1)]+=g;
}
printf("%d\n",f[n][0]);
return 0;
}

解决办法:一次转移

即通过dfs完成转移,做到不重不漏

在一个状态下dfs下一排可能的所有状态

具体见ppt

// It is made by XZZ
#include<cstdio>
#include<algorithm>
#define lb(a) (a&-a)
#define Fname "floor"
using namespace std;
#define rep(a,b,c) for(rg int a=b;a<=c;a++)
#define drep(a,b,c) for(rg int a=b;a>=c;a--)
#define erep(a,b) for(rg int a=fir[b];a;a=nxt[a])
#define il inline
#define rg register
#define vd void
typedef long long ll;
il int gi(){
rg int x=0;rg char ch=getchar();
while(ch<'0'||ch>'9')ch=getchar();
while(ch>='0'&&ch<='9')x=x*10+ch-'0',ch=getchar();
return x;
}
ll f[12][1<<11],n,m;
il vd dfs(const ll&F,const int&i,const int&j,int x,int now){
if(now==m)f[i+1][x]+=F;
else if(j&(1<<now))dfs(F,i,j,x,now+1);//已经有了
else{
dfs(F,i,j,x|(1<<now),now+1);//竖着
if((now!=m-1)&&!(j&(1<<now+1)))dfs(F,i,j,x,now+2);//横着
}
}
int main(){
#ifdef xzz
freopen(Fname".in","r",stdin);
freopen(Fname".out","w",stdout);
#endif
n=gi(),m=gi();
if((n*m)&1){puts("0");return 0;}
f[0][0]=1;
rep(i,0,n-1)rep(j,0,(1<<m)-1)dfs(f[i][j],i,j,0,0);
printf("%lld\n",f[n][0]);
return 0;
}

其实还可以有更快的:每个j转移过去的都相同,可以先预处理每个j转移的,用邻接表实现,应该快的飞起

虽然我懒得写了(逃


硬木地板

举行计算机科学家盛宴的大厅的地板为 M×N (1<=M<=9, 1<=N<=9)的矩形。现在必须要铺上硬木地板砖。可以使用的地板砖形状有两种:

  1. 2×1 的矩形砖
  2. 2×2 中去掉一个 1×1 的角形砖

    你需要计算用这些砖铺满地板共有多少种不同的方案。

    注意:必须盖满,地板砖数量足够多,不能存在同时被多个板砖覆盖的部分。

    输入数据

    包含 M 和 N。

    输出数据

    输出方案总数,如果不可能那么输出 0 。

    样例

    输入:floor.in

    2 3

    输出:floor.out

    5

会上面那个基本就会这个了额。。。

改一下dfs就好了

// It is made by XZZ
#include<cstdio>
#include<algorithm>
#define Fname "floor2"
using namespace std;
#define rep(a,b,c) for(rg int a=b;a<=c;a++)
#define drep(a,b,c) for(rg int a=b;a>=c;a--)
#define erep(a,b) for(rg int a=fir[b];a;a=nxt[a])
#define il inline
#define rg register
#define vd void
typedef long long ll;
il int gi(){
rg int x=0;rg char ch=getchar();
while(ch<'0'||ch>'9')ch=getchar();
while(ch>='0'&&ch<='9')x=x*10+ch-'0',ch=getchar();
return x;
}
ll f[10][1<<9],n,m;
il vd dfs(const ll&F,const int&i,const int&j,int x,int now){
if(now==m)f[i+1][x]+=F;
else if(j&(1<<now))dfs(F,i,j,x,now+1);
else{
if(!(x&(1<<now))){
dfs(F,i,j,x|(1<<now),now+1);//竖着
if(now&&(!(x&(1<<now-1))))dfs(F,i,j,x|(1<<now)|(1<<now-1),now+1);//┘
if(now!=m-1)dfs(F,i,j,x|(1<<now)|(1<<now+1),now+1);//└
}
if((now!=m-1)&&!(j&(1<<now+1))){
dfs(F,i,j,x,now+2);//横着
if(!(x&(1<<now)))dfs(F,i,j,x|(1<<now),now+2);//┌
dfs(F,i,j,x|(1<<now+1),now+2);//┐
} }
}
int main(){
#ifdef xzz
freopen(Fname".in","r",stdin);
freopen(Fname".out","w",stdout);
#endif
n=gi(),m=gi();
f[0][0]=1;
rep(i,0,n-1)rep(j,0,(1<<m)-1)dfs(f[i][j],i,j,0,0);
printf("%lld\n",f[n][0]);
return 0;
}

PS.具体参见上面ppt链接

dp合集 广场铺砖问题&&硬木地板的更多相关文章

  1. 9.15 DP合集水表

    9.15 DP合集水表 显然难了一些啊. 凸多边形的三角剖分 瞄了一眼题解. 和蛤蛤的烦恼一样,裸的区间dp. 设f[i][j]表示i~j的点三角剖分最小代价. 显然\(f[i][i+1]=0,f[i ...

  2. 9.14 DP合集水表

    9.14 DP合集水表 关键子工程 在大型工程的施工前,我们把整个工程划分为若干个子工程,并把这些子工程编号为 1. 2. --. N:这样划分之后,子工程之间就会有一些依赖关系,即一些子工程必须在某 ...

  3. 【CJOJ2498】【DP合集】最长上升子序列 LIS

    题面 Description 给出一个 1 ∼ n (n ≤ 10^5) 的排列 P 求其最长上升子序列长度 Input 第一行一个正整数n,表示序列中整数个数: 第二行是空格隔开的n个整数组成的序列 ...

  4. CJOJ 【DP合集】最长上升序列2 — LIS2

    题面 已知一个 1 ∼ N 的排列的最长上升子序列长度为 K ,求合法的排列个数. 好题(除了我想不出来我应该找不到缺点), 想一想最长上升子序列的二分做法, 接在序列后面或者替换. 所以对于每一个位 ...

  5. 【DP合集】tree-knapsack

    Description 给出一个 N 个节点的有根树,点编号 1 ∼ N ,编号为 i 的点有权值 v i .请选出一个包含树根的,点数 不超过 K 的连通块,使得点权和最大. Input 输入的第一 ...

  6. 【DP合集】m-knapsack

    给出 n 个物品,第 i 个物品有重量 w i .现在有 m 个背包,第 i 个背包的限重为 c i ,求最少用几个背 包能装下所有的物品. Input 输入的第一行两个整数 n, m ( n ≤ 2 ...

  7. 【DP合集】背包 bound

    N 种物品,第 i 种物品有 s i 个,单个重量为 w i ,单个价值为 v i .现有一个限重为 W 的背包,求能容 纳的物品的最大总价值. Input 输入第一行二个整数 N , W ( N ≤ ...

  8. 【DP合集】合并 union

    给出一个 1 ∼ N 的序列 A ( A 1 , A 2 , ..., A N ) .你每次可以将两个相邻的元素合并,合并后的元素权值即为 这两个元素的权值之和.求将 A 变为一个非降序列,最少需要多 ...

  9. 【DP合集】棋盘 chess

    给出一张 n × n 的棋盘,格子有黑有白.现在要在棋盘上放棋子,要求: • 黑格子上不能有棋子 • 每行每列至多只有一枚棋子 你的任务是求出有多少种合法的摆放方案.答案模 109+7109+7 . ...

随机推荐

  1. TP,TN,FP,FN

    一张图搞定~~~ [转]https://blog.csdn.net/u011956147/article/details/78967145

  2. windows下使用Git

    如何在windows下使用Git? 通过这里下载Git bash,你就可以像在Linux命令行一样操作git工具. 进入Git bash环境,默认是在当前用户路径下. 在Linux下,我们有根目录,在 ...

  3. Golang测试包

    Golang测试包 golang自带了测试包(testing),直接可以进行单元测试.性能分析.输出结果验证等.简单看着官方文档试了试,总结一下: 目录结构和命令 使用golang的测试包,需要遵循简 ...

  4. AttributeError: 'module' object has no attribute get'

    最近在写python requests相关内容易,突然报错AttributeError: 'module' object has no attribute 'get'" 脚本肯定没问题 怎么 ...

  5. shell一次性执行多条命令

    1.每个命令之间用;隔开说明:各命令的执行给果,不会影响其它命令的执行.换句话说,各个命令都会执行,但不保证每个命令都执行成功. 2.每个命令之间用&&隔开说明:若前面的命令执行成功, ...

  6. .Net core 使用NPOI 直接导入Excel到数据库(即不先将Excel保存到服务器再读取文件到数据库)

    /// <summary> /// 导入信息 /// </summary> /// <param name="file"></param& ...

  7. Java解析Excel之应用Reflection等技术实现动态读取

    目录树 背景 技术选型 问题分析 技术要点及难点分析 源码分析 测试用例 背景 Tip:因为产品提的需求我都开发完了,进行了项目提测:前天老大走过来说:你用spring-boot开发一个解析Excel ...

  8. 项目中常用的MySQL优化方法--壹拾玖条

    1.EXPLAIN 做MySQL优化,我们要善用EXPLAIN查看SQL执行计划. 下面来个简单的示例,标注(1.2.3.4.5)我们要重点关注的数据: type列,连接类型.一个好的SQL语句至少要 ...

  9. NOIP模拟赛D2T1自己的解题思路

    T1题目在此: 数轴上有n个球,每个球直径为1,第 ii 个球的左端点为pi即占据了数轴上[pi,pi+1][pi,pi+1]).在 P位置有一堵墙.有q个操作,每次要么以x位置为左端点放一个新球(如 ...

  10. C# 删除指定目录下的所有文件及文件夹

    using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Tex ...