棋盘问题
Time Limit: 1000MS | Memory Limit: 10000K | |
Total Submissions: 44012 | Accepted: 21375 |
Description
在一个给定形状的棋盘(形状可能是不规则的)上面摆放棋子,棋子没有区别。要求摆放时任意的两个棋子不能放在棋盘中的同一行或者同一列,请编程求解对于给定形状和大小的棋盘,摆放k个棋子的所有可行的摆放方案C。
Input
输入含有多组测试数据。
每组数据的第一行是两个正整数,n k,用一个空格隔开,表示了将在一个n*n的矩阵内描述棋盘,以及摆放棋子的数目。 n <= 8 , k <= n
当为-1 -1时表示输入结束。
随后的n行描述了棋盘的形状:每行有n个字符,其中 # 表示棋盘区域, . 表示空白区域(数据保证不出现多余的空白行或者空白列)。
每组数据的第一行是两个正整数,n k,用一个空格隔开,表示了将在一个n*n的矩阵内描述棋盘,以及摆放棋子的数目。 n <= 8 , k <= n
当为-1 -1时表示输入结束。
随后的n行描述了棋盘的形状:每行有n个字符,其中 # 表示棋盘区域, . 表示空白区域(数据保证不出现多余的空白行或者空白列)。
Output
对于每一组数据,给出一行输出,输出摆放的方案数目C (数据保证C<2^31)。
Sample Input
2 1
#.
.#
4 4
...#
..#.
.#..
#...
-1 -1
Sample Output
2
1
Source
题目链接:http://poj.org/problem?id=1321
分析:
这个题目的大意是给定一个棋盘和给定我们需要摆放的棋子的数目,然后问我们有几种摆放方式。首先我们可以明确这是一个深度搜索的题目,与八皇后问题相似。我们建立一个函数DFS用来累计可行的方案数,我们走过一列我们就把它标记下来下次的时候就不可以再摆放在这一列(因为题目要求不可以将棋子摆放在同一行和同一列)
然后就从下一行开始寻找可行的地方,直到我们摆放的棋子数与我们被要求摆放的棋子数相同时,我们就将方案数进行一次++,然后在进行递归下去。
DFS板子题,还在熟练中,争取达到闭着眼睛三分钟敲出板子!
此题我每一步给出详细解释,新手学习,大神见谅!
下面给出详解代码:
#include<iostream>
#include <algorithm>
#include <stdio.h>
#include <string.h>
using namespace std;
int visit[];
char mp[][];
int ans;//ans表示方案数
int k;//k表示棋子数目
int n;//n表示棋盘的大小
int DFS(int x,int y)
{
if(y>=k)//判断是否棋子已经用完,如果用完,记录方案数加1,然后直接返回0
{
ans++;
return ;
}
for(int i=x;i<n;i++)
{
for(int j=;j<n;j++)
{
if(!visit[j]&&mp[i][j]=='#')//标记数组仅仅标记某一列上是否有棋子,因为每次递归下一列,所以每一列不会有冲突,只需判断这一列上是否有其他棋子
{
visit[j]=true;//如果该位置该列没被标记且为棋盘,那么在这里放上棋子,并标记,
DFS(i+,y+);//搜索下一列
visit[j]=false;//还要注意修改标记后递归回来要及时复原
}
}
}
return ;
}
int main()
{
while(cin>>n>>k)
{
if(n==-&&k==-)
break;
memset(visit,false,sizeof(visit));
memset(mp,false,sizeof(mp));
for(int i=;i<n;i++)
cin>>mp[i];
ans=;
DFS(,);
cout<<ans<<endl;
}
return ;
}
多年以后重写此题,想出了另外一种解决办法!
题目意思很明了,其中'#'可以放棋子,'.'不能,并且同一行或同一列不能放两个棋子,对于数据一游两种放法('*'代表放的棋子)
*.
#.
或
#.
.*
对于数据二只有一种情况
...*
..*.
.*..
*...
这题只需要深搜,每次从上一个放棋子地方的下一行开始寻找可以放棋子的地方,
当发现该点时,记录行数,并更新棋盘,将于此点同行同列的都更新为'.',
如果找不到,则返回,当把所有棋子都放上去的时候,则找到一个接,计数+1,就这样进行搜索,可以保证AC
#include <iostream>
using namespace std;
struct p{
char s[][];//棋盘
int beforerow;//上一个棋子的行数
};
//st表示开始搜索的棋子所在的那一行,resnum表示剩余可放的棋子数
int n,resnum;//n表示当前的棋盘大小为n*n,k表示可放的总棋子数
int ans;//摆放的所有可能数
void DFS(p temp,int resnum)
{
if(resnum==)//如果剩余棋子数量等于0,说明所有棋子都已经放好了,答案数+1返回
{
ans++;
return;
}
//否则就得从当前棋子的下一行开始搜索
//并且我们知道棋子数k大于行数n的情况显然是不存在的,有了肯定是无解情况,这里就不需要讨论这个问题
int i,j;
for(i=temp.beforerow+;i<=n-resnum;i++)
{
for(j=;j<n;j++)
{
if(temp.s[i][j]=='#')
{
p temp1;
temp1=temp;
temp1.beforerow=i;//记下改点的行数
int k;
for(k=i+;k<n;k++)//更新棋盘
{
temp1.s[k][j]='.';
}
DFS(temp1,resnum-);//放好一个棋子继续搜
}
}
}
}
int main()
{
while(cin>>n>>resnum)
{ if(n==-&&resnum==-)
break;
ans=;
p temp;
temp.beforerow=-;//此时还未放棋子,初始化为-1
for(int i=;i<n;i++)
{
cin>>temp.s[i];
}
DFS(temp,resnum);
cout<<ans<<endl;
}
return ;
}