状压DP入门

状态压缩:现在有一些物品,若我们选择某件物品,则这件物品为1,不选则为0。那么对于这些物品的选择的状态可以表示为0 0 0 1 1。这个表示选择第1件物品和第2件物品,其余的不选。对于所有状态都可以这样表示。而这些状态我们可以把他看成一个10进制的数。如例子中的状态,就可以表示为3,3的二进制为00011,这样就把状态压缩了。不需要用数组去表示是否选择物品。

例题:有一个N*M(N<=5,M<=1000)的棋盘,现在有1*2及2*1的小木块无数个,要盖满整个棋盘,有多少种方式?答案只需要mod1,000,000,007即可。

我们通过找一些比较小的数据会发现一个特性。某一列的摆放个数只会收到上一列的影响(某一列的上一列之前全部摆放好了)。也就是说第三列只会收到第二列的影响,第一列影响不了第三列。所以我们只需找某相邻两列的状态之间的练习即可。
我们会发现,N < =5 ,代表列数非常短,也就是说某一列的状态数很小。这就给了我们用2进制的数去枚举状态数的启发。
dp【i】【j】表示当在i列时,状态为j时的摆放方式数。
dp【i+1】【j】 = 第i列中所有能到达dp【i+1】【j】的摆放数之和。
递推方程很好找。具体一些实现细节看代码。
#include<cstdio>
#include<cstring>
#include<iostream>
using namespace std;

const int  mod = 1000000007;
int dp[1005][50]; // 因为 n小于5所以状态数不会超过50  
int n,m;

void dfs(int i,int j,int state,int next){   // i 表示 当前在摆放第i列 , j 表示 在摆放第 j 行 ,state 表示第i列当前状态, next表示对下一列的影响 
    if(j==n){ // 若对摆放i列结束。 
        dp[i+1][next]+=dp[i][state]; // 所有可以到达 next状态 的摆放数 之和 
        dp[i+1][next]%=mod; 
        return;
    }
    if(((1<<j)&state)>0) // 某个位置不为空 跳过 
		dfs(i,j+1,state,next);
    if(((1<<j)&state)==0)  // 某个位置为空 放一个 1*2的 
		dfs(i,j+1,state,next|(1<<j));
	if(j+1<n&&((1<<j)&state)==0&&((1<<(j+1)&state)==0))  // 同一列 某个位置 为空 切 下一行的位置也为空 放一个 2*1; 
		dfs(i,j+2,state,next);
    return;
}
int main(){
    cin>>n>>m;
    memset(dp,0,sizeof(dp)); // 初始化 
    dp[1][0]=1; // dp【1】【0】这种情况肯定为0,因为只有什么都不放这一种情况。 
    for(int i=1;i<=m;++i) // 遍历m列 
      for(int j=0;j<(1<<n);++j) // 遍历所有种状态 。 
            if(dp[i][j])  dfs(i,0,j,0);
    cout<<dp[m+1][0]<<endl; // 这是遍历到n*m的情况 
    return 0;
 }
还有不懂私聊博主。


  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值