Chinaunix首页 | 论坛 | 博客
  • 博客访问: 342928
  • 博文数量: 122
  • 博客积分: 5000
  • 博客等级: 大校
  • 技术积分: 1191
  • 用 户 组: 普通用户
  • 注册时间: 2009-10-24 11:12
文章分类

全部博文(122)

文章存档

2010年(122)

我的朋友

分类: C/C++

2010-04-11 20:40:20

一、问题描述

Description

在一个给定形状的棋盘(形状可能是不规则的)上面摆放棋子,棋子没有区别。要求摆放时任意的两个棋子不能放在棋盘中的同一行或者同一列,请编程求解对于给定形状和大小的棋盘,摆放k个棋子的所有可行的摆放方案C

Input

输入含有多组测试数据。
每组数据的第一行是两个正整数,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

 

二、解题思路

使用回溯法求解。首先将棋盘的所有空白区坐标放在一个数组p中,问题等同于在p个中选择k个位置放棋子,当然同行同列不能有多于2个。row[]col[]分别表示该行或该列是否已经放过棋子。对于每个位置有放与不放两个选择,解空间是一棵满二叉树,深度为NN为空白区的个数),当放满k个时,计数器c+=1

三、代码

 

#include<iostream>
using namespace std;
struct Pos
{
    int x;
    int y;
};
Pos p[81];//空白棋子的位置
int N;

int k;
int n;
bool col[9];
bool row[9];
int c=0;
void backtrace(int n,int j)
{
    int i;
    if(n>=k)
    {
        c++;
        return ;
    }
    for(i=j;i<N;++i)
    {
        int x=p[i].x;
        int y=p[i].y;
        if(row[x] && col[y] )
        {
            row[x]=false;
            col[y]=false;
            backtrace(n+1,i+1);
            row[x]=true;
            col[y]=true;
        }
    }
}
int main()
{
    int i,j;
    char ch;
    while(cin>>n>>k)
    {
        if(n==-1 && k==-1)
            break;
        N=0;
        c=0;
        memset(row,0,sizeof(row));
        memset(col,0,sizeof(col));
        for(i=1;i<=n;++i)
        {
            for(j=1;j<=n;++j)
            {
                cin>>ch;
                if(ch=='#')
                {
                    p[N].x=i;
                    p[N].y=j;
                    row[i]=true;
                    col[j]=true;
                    N++;
                }
            }
        }
        backtrace(0,0);
        cout<<c<<endl;
    }
    return 0;
}


阅读(2130) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~