• HDOJ2553-N皇后问题(DFS)


     

    N皇后问题

    Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
    Total Submission(s): 12279    Accepted Submission(s): 5535


    Problem Description
    在N*N的方格棋盘放置了N个皇后,使得它们不相互攻击(即任意2个皇后不允许处在同一排,同一列,也不允许处在与棋盘边框成45角的斜线上。
    你的任务是,对于给定的N,求出有多少种合法的放置方法。

     
    Input
    共有若干行,每行一个正整数N≤10,表示棋盘和皇后的数量;如果N=0,表示结束。
     
    Output
    共有若干行,每行一个正整数,表示对应输入行的皇后的不同放置数量。
     
    Sample Input
    1 8 5 0
     
    Sample Output
    1 92 10
     
    Author
    cgf
     
    Source
    知识点:DFS 回溯法(可见http://www.cnblogs.com/ZP-Better/p/4649694.html)
    题意:见题目描述(obviously)
    思路:DFS 先找状态(目前所在第几行,目前所在第几列,目前已经放置的皇后数量)扩展方式:从第一列到最后一列搜索到合适的位置,最终状态:目前放置的皇后数量等于题目所要求的皇后数量。
    重点:1.如何判断2个皇后不允许不允许处在与棋盘边框成45角的斜线上。将棋盘看成坐标图,row1+column1=row2+column2=c(x1+y1=x2+y2=c) or row1-column1=row2-column2=c(x1-y1=x2-y2=c)来判断2个皇后是否在棋盘边框成45角的斜线上。为了满足这个条件还需要将目前这个点与已经放置过的点(需要遍历前面的点)判断才满足。
            2.这题卡时间,又由于数字较小所以要打表将这10种情况用数组存下。
     1 #include<cstdio>
     2 #include<cstring>
     3 #include<iostream>
     4 using namespace std;
     5 int n,cnt;
     6 int vis[11];//标记数组
     7 int picture[11];//用下标来表示行号,自身存储列号。
     8 bool judge(int r,int r2,int c2)//判断2个皇后是否在棋盘边框成45角的斜线上
     9 {
    10     for(int i=0;i<=r;i++)//遍历已经放置皇后的点
    11     {
    12     if(i+picture[i]==r2+c2||i-picture[i]==r2-c2)
    13         return false;
    14     }
    15     return true;
    16 }
    17 void dfs(int row,int column,int num)
    18 {
    19     if(num==n)
    20     {
    21     cnt++;
    22     return;
    23     }
    24     for(int i=0;i<n;i++)
    25         if(!vis[i]&&judge(row,row+1,i))
    26         {
    27             picture[row+1]=i;
    28             //printf("%dorz%dorz%d
    ",row+1,i,num);
    29             vis[i]=1;
    30             dfs(row+1,i,num+1);
    31             vis[i]=0;
    32         }
    33 }
    34 int main()
    35 {
    36     while(~scanf("%d",&n))
    37     {
    38     cnt=0;
    39     memset(vis,0,sizeof(vis));
    40     memset(picture,0,sizeof(picture));
    41     for(int i=0;i<n;i++)
    42     {
    43      vis[i]=1;
    44      picture[0]=i;
    45      dfs(0,i,1);
    46      vis[i]=0;//回溯法:还原标记
    47     }
    48     printf("%d
    ",cnt);
    49     }
    50     return 0;
    51 }

     通过上面代码得到10种情况答案。下面是AC代码

     1 #include <cstdio>
     2 #include <iostream>
     3 int a[100]={1,1,0,0,2,10,4,40,92,352,724};
     4 int n;
     5 int main()
     6 {
     7   while(~scanf("%d",&n))
     8   {if(n==0)
     9   break;
    10   printf("%d
    ",a[n]);}
    11   return 0;
    12 }
  • 相关阅读:
    在Dictionary使用foreach的注意
    [公告]新增项目交流区
    博客园Blog程序中又一个奇怪问题
    公告
    网络爬客光顾博客园
    爬网的影响
    致歉
    [业界新闻]中文IT百科Beta版于2006年7月10日正式上线
    公告
    疯狂的爬网
  • 原文地址:https://www.cnblogs.com/ZP-Better/p/4649657.html
Copyright © 2020-2023  润新知