动态规划:铺砖问题

问题描述:给定n*m的格子,每个格子被染成了黑色或者白色。现在要用1*2的砖块覆盖这些格子,要求块与块之间互相不重叠,且覆盖了所有白色的格子,但不覆盖任意一个黑色格子。求一共有多少种覆盖方法,输出方案数对M取余后的结果。

限制条件:

  • 1 <= n <= 15
  • 1 <= m <= 15
  • 2 <= M <= 10^9

样例输入:

n = 3

m = 3

...(.代表白色,x带表黑色)

.x.

...

输出:2

解题思路:

为了不重复统计,我们每次从最左上方的空格处开始放置。对于哪些格子已经被覆盖过了,只需要用一个bool数组来维护就可以了。首先,黑色的格子不能被覆盖,因此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个格子通过状态压缩编码进行记忆话搜索。

代码实例:

int dp[2][1 << MAX_M];
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 ++){
				if((used >> j &1) || color[i][j]){
					next[used] = crt[used &~(1<<j)];
				}else{
					int res = 0;
					if(j + 1 > m && !(used >> (j+1) & 1) && !color[i][j+1]){
						res += crt[used | 1 << (j+1)];
					}
					if(i+1 < n && !color[i+1][j]){
						res += crt[used | 1<<j];
					}
					next[used] = res % M;
				}
			}
			swap(crt,next);
		}
	}
	printf("%d\n",crt[0]);
}

 

posted @ 2018-07-28 11:19  Dr_Lo  阅读(830)  评论(0编辑  收藏  举报