【暑假集训专题#搜索 POJ 1321】

题目链接click here~~

题目大意

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

Sample Input

2 1

#.

.#

4 4

...#

..#.

.#..

#...

-1 -1

Sample Output

2

解题思路】:

在别人的博客看到了,和作者的基本一样,写的比较清楚,直接复制过来了:

这道题目类似N皇后问题,与之不同的是每一行不一定有棋盘,所以dfs里要注意不一定是当前行。思路很简单,只需从第一行第一个开始搜索,如果该位置该列没被标记且为棋盘,那么在这里放上棋子,并标记,因为每行每列不能冲突,所以搜索下一行,比并且棋子数加1。每次搜索之前先要判断是否棋子已经用完,如果用完,记录方案数加1,然后直接返回。直到所有搜索全部完成,此时已得到全部方案数。此题还需注意标记数组仅仅标记某一列上是否有棋子,因为每次递归下一行,所以每一行不会有冲突,只需判断这一列上是否有其他棋子。还要注意修改标记后递归回来要及时复原。

代码:

#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include <set>
#include <queue>
#include <string.h>
#include <iostream>
#include <algorithm>
using namespace std;
const int N=20;
int dir[4][2]= {{1,0},{-1,0},{0,1},{0,-1}};
int num,line,res;
char mat[N][N];
bool vis_row[N];
void dfs(int _line,int now_num)
{
    if(now_num==num){///棋子全部放置
        ++res;
        return;
    }
    for(int i=_line; i<line; ++i){
        for(int j=0; j<line; ++j)
            if(mat[i][j]=='#'&&!vis_row[j]){
                vis_row[j]=true;
                dfs(i+1,now_num+1);
                vis_row[j]=false;
            }
    }
}
int main()
{
    while(~scanf("%d%d",&line,&num)&&line!=-1&&num!=-1)
    {
        memset(vis_row,false,sizeof(vis_row));
        for(int i=0; i<line; i++) scanf("%s",mat[i]);
        res=0;
        dfs(0,0);   ///按行搜索,从0行递增;当前放置的棋子
        printf("%d\n",res);
    }
    return 0;
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-11-05 17:19:51

【暑假集训专题#搜索 POJ 1321】的相关文章

【暑假集训专题#搜索】

poj 2386 Lake Counting [题意]: 有一个大小为N×M的园子.雨后积起了水. 八连通的积水被觉得是连接在一起的.请求出园子里总共同拥有多少水洼?(八连通指的是下图中相对W 的*的部分) Sample Input 10 12 W........WW. .WWW.....WWW ....WW...WW. .........WW. .........W.. ..W......W.. .W.W.....WW. W.W.W.....W. .W.W......W. ..W.......

【暑假集训专题#搜索 HDU1241】

[题目链接]click here~~ [题目大意]'@'代表油田位置,'*'代表地面,八个方向相邻的油田视为一个,求给定地图里油田数目 [解题思路]八个方向搜索即可 代码: #include <stdio.h> #include <string.h> #include <iostream> #include <algorithm> using namespace std; const int N=1010; int dir4[4][2]= {{1,0},{0

POJ 1321 棋盘问题(搜索)

题目链接:POJ 1321 棋盘问题 Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 28825   Accepted: 14276 Description 在一个给定形状的棋盘(形状可能是不规则的)上面摆放棋子,棋子没有区别.要求摆放时任意的两个棋子不能放在棋盘中的同一行或者同一列,请编程求解对于给定形状和大小的棋盘,摆放k个棋子的所有可行的摆放方案C. Input 输入含有多组测试数据. 每组数据的第一行是两个正整数,

暑假集训之专题----拓扑排序题解

第一单: Problem A Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 32768/32768K (Java/Other) Total Submission(s) : 26   Accepted Submission(s) : 5 Problem Description ACM-DIY is a large QQ group where many excellent acmers get together. It is so h

2017暑假集训前总结和规划

距离大一进来已经一年了啊,感觉还是啥也不会,哎,太差了,总结一下这一年都学了写什么吧! 大一寒假开始专题,刷过的有:dp,dfs和bfs,数论(gcd拓展gcd,欧拉定理等等,但是中国剩余定理没学,等复习的时候再学吧),并查集,最短路(bellman-fprd,dijkstra,floyd-warshall,spfa),最小生成树(prim,kruskal),线段树,二分三分 大一下学期有:拓扑排序,基础计算几何(直线线段相交,快速排除实验,跨立实验),矩阵快速幂,博弈基础(nim博弈,威佐夫博

暑假集训0808

来HIT参加暑假集训也有将近一周了,一直什么都没写= =. 记一下今天的比赛吧,以后争取每天更新一篇总结. 我是弱比.只能出6题. A:POJ1417 很容易发现yes表示两个人是同一类,no是不同类,然后怎么判断方案是否唯一我就不会了...我是有多么的弱...类似于背包DP,就是用前i个大类填满j个人数的方案. B:HDU1711 裸KMP C:POJ 2503 一个字典的实现.很显然的Trie.(我就想说出题人就不能给个n么..读入麻烦死..) D:POJ2828  比赛的时候还是想的有问题

CSU-ACM2014暑假集训基础组训练赛(1) 解题报告

•Problem A HDU 4450                 水题,签到题 水题..没啥好说的.给大家签到用的. 1 #include <cstdio> 2 int main(){ 3 int n,a,ans; 4 while(scanf("%d",&n),n){ 5 ans = 0; 6 for(int i = 0;i < n;i++){ 7 scanf("%d",&a); 8 ans += a*a; 9 } 10 pr

「总结」2019暑假集训

啊,我最喜欢的暑假集训终究还是结束了. 感觉集训收获的还是挺大的,不管是在知识方面还是心态方面,感觉现在考试心态稳了很多,不管是考前考时考后,都可以很快的调整了.大概就是教练所说的考试心态调整的加速.最近感觉非常好,虽然水题还是老爆零,考得也不怎么样,不过我的确是飞快的在进步了,只要我在进步就好了,我很心满意足的. 总结一下接近30场的考试吧. 一开始的7场一直只有50分左右,而且还有两次没有交卷子,气得我把纸贴在电脑上提醒自己交卷子,虽然成绩并不怎么样,不过我还好在也不犯这个错误了. 然后就理

[暴力搜索] POJ 3087 Shuffle&#39;m Up

Shuffle'm Up Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 10003   Accepted: 4631 Description A common pastime for poker players at a poker table is to shuffle stacks of chips. Shuffling chips is performed by starting with two stacks o