POJ
来源:互联网 发布:西安软件服务业产值 编辑:程序博客网 时间:2024/06/02 02:00
在一个给定形状的棋盘(形状可能是不规则的)上面摆放棋子,棋子没有区别。要求摆放时任意的两个棋子不能放在棋盘中的同一行或者同一列,请编程求解对于给定形状和大小的棋盘,摆放k个棋子的所有可行的摆放方案C。
输入含有多组测试数据。
每组数据的第一行是两个正整数,n k,用一个空格隔开,表示了将在一个n*n的矩阵内描述棋盘,以及摆放棋子的数目。 n <= 8 , k <= n
当为-1 -1时表示输入结束。
随后的n行描述了棋盘的形状:每行有n个字符,其中 # 表示棋盘区域, . 表示空白区域(数据保证不出现多余的空白行或者空白列)。
每组数据的第一行是两个正整数,n k,用一个空格隔开,表示了将在一个n*n的矩阵内描述棋盘,以及摆放棋子的数目。 n <= 8 , k <= n
当为-1 -1时表示输入结束。
随后的n行描述了棋盘的形状:每行有n个字符,其中 # 表示棋盘区域, . 表示空白区域(数据保证不出现多余的空白行或者空白列)。
对于每一组数据,给出一行输出,输出摆放的方案数目C (数据保证C<2^31)。
2 1#..#4 4...#..#..#..#...-1 -1
21
一个中文水体,从提议就可以知道是个暴力DFS,但是我还是太水了,只知道是个dfs真的不知道怎么敲出来,因为递归一直理解很模糊,所以最近一直想做一些DFS的题,所以先从这个大水题开始做吧,看了看不同大佬的博客才懂了。
思路:从第一行开始遍历,如果这个地方可以放棋子,那么标记下,遍历下面的,如果当放的棋子达到要求个数就直接return,用一个全局变量去保存符合的方式。
代码如下:
#include <cstdio>#include <algorithm>#include <iostream>#include <cstring>using namespace std;char s[30][30];bool vis[30];int sum, m, n;void dfs(int x, int num){ if(num == n) { sum++; return; } else { for(int i = x; i < m; i++) // 遍历从这行开始是否有符合条件的# for(int j = 0; j < m; j++) if(s[i][j] == '#' && !vis[j]) { vis[j] = 1; dfs(i + 1, num + 1); vis[j] = 0; }}}int main(){ while(scanf("%d%d", &m, &n) != EOF) { memset(s, 0, sizeof(s)); memset(vis, 0, sizeof(vis)); sum = 0; getchar(); if(m == -1 && n == -1) break; for(int i = 0; i < m; i++) scanf("%s", s[i]); dfs(0, 0); //从第一行开始遍历 cout << sum << endl; } return 0;}
0 0