题目描述
在一个给定形状的棋盘上面摆放棋子,棋子没有区别。
要求摆放时任意的两个棋子不能放在棋盘中的同一行或者同一列,
请编程求解对于给定形状和大小的棋盘,摆放 k 个棋子的所有可行的摆放方案 C。
输入格式
输入含有多组测试数据。
每组数据的第一行是两个正整数 n, k,以空格隔开,表示将在一个 n×n 的矩阵内描述棋盘,以及摆放棋子的数目。
当为 −1 −1
时表示输入结束。
随后的 n 行描述了棋盘的形状:每行有 n 个字符,其中 #
表示棋盘区域,.
表示空白区域
数据保证不出现多余的空白行或者空白列
输出格式
对于每一组数据,给出一行输出,输出摆放的方案数目 C(数据保证C < 231)。
输入样例
2 1
#.
.#
4 4
...#
..#.
.#..
#...
-1 -1
输出样例
2
1
数据范围
k ≤ n ≤ 8
题解
DFS:
解题思路
:
- 由于棋子之间没有任何区别,那么先放在第 1 行再放到第 2 行和先放在第 2 行再放到第 1 行属于同种方案;
- 因此,可以规定一种搜索顺序,即从上往下依次搜索;
#include <iostream>
#include <cstring>
using namespace std;
const int N = 10;
bool col[N];
char g[N][N];
int n, k, ans;
void dfs(int k, int start)
{
if(!k)
{
ans ++;
return;
}
for (int i = start; i < n; i ++)
for (int j = 0; j < n; j ++)
{
if(g[i][j] == '.' || col[j]) continue;
col[j] = true;
dfs(k - 1, i + 1);
col[j] = false;
}
}
int main()
{
while(cin >> n >> k, n != -1 && k != -1)
{
ans = 0;
memset(col, false, sizeof col);
for (int i = 0; i < n; i ++) cin >> g[i];
dfs(k, 0);
cout << ans << endl;
}
return 0;
}