在一个给定形状的棋盘(形状可能是不规则的)上面摆放棋子,棋子没有区别。
要求摆放时任意的两个棋子不能放在棋盘中的同一行或者同一列,请编程求解对于给定形状和大小的棋盘,摆放 k 个棋子的所有可行的摆放方案数目 C。
输入格式
输入含有多组测试数据。
每组数据的第一行是两个正整数 n,k,用一个空格隔开,表示了将在一个 n∗n 的矩阵内描述棋盘,以及摆放棋子的数目。当为
-1 -1
时表示输入结束。随后的 n 行描述了棋盘的形状:每行有 n 个字符,其中
#
表示棋盘区域,.
表示空白区域(数据保证不出现多余的空白行或者空白列)。输出格式
对于每一组数据,给出一行输出,输出摆放的方案数目 C(数据保证 C<2^31)。
数据范围
n≤8,k≤n
输入样例:
2 1 #. .# 4 4 ...# ..#. .#.. #... -1 -1
输出样例:
2 1
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
#include <cmath>
#include <queue>
#include <set>
#include <stack>
using namespace std;
//#define int long long
typedef pair<int,int>PII;
constexpr int N=5e4+7;
char s[10][10];
bool r[10];
int n,m;
int res;
int cnt;
void dfs(int x){
if(cnt==m){
res++;
return;
}
if (x > n) return;
for(int i=1;i<=n;i++){
if(s[i][x]=='#'&&!r[i]){
r[i]= true;
cnt++;
dfs(x+1);
cnt--;
r[i]= false;
}
}
dfs(x+1);
}
signed main()
{
ios_base::sync_with_stdio(0), cin.tie(0), cout.tie(0);
while(cin>>n>>m&&n!=-1&&m!=-1){
res=0,cnt=0;
memset(r,0,sizeof r);
for(int i=1;i<=n;i++){
for(int j=1;j<=n;j++){
cin>>s[i][j];
}
}
dfs(1);
cout<<res<<endl;
}
return 0;
}