递归--八皇后

我不是女神ヾ 2021-12-17 02:59 416阅读 0赞

  八皇后这个典的问题,是每个真正程序员必须经历过的。这也是我第二次来解决这个问题了,第一次应该是学数据结构那时候吧。这次写起来顺利多了,基本没遇到什么卡壳的地方。递归+回溯。

问题描述
会下国际象棋的人都很清楚:皇后可以在横、竖、斜线上不限步数地吃掉其他棋子。如
何将8 个皇后放在棋盘上(有8 * 8 个方格),使它们谁也不能被吃掉!这就是著名的八皇后
问题。 对于某个满足要求的8 皇后的摆放方法,定义一个皇后串a 与之对应,即a=b1b2…b8,
其中bi 为相应摆法中第i 行皇后所处的列数。已经知道8 皇后问题一共有92 组解(即92 个
不同的皇后串)。给出一个数b,要求输出第b 个串。串的比较是这样的:皇后串x 置于皇
后串y 之前,当且仅当将x 视为整数时比y 小。
输入数据
第 1 行是测试数据的组数n,后面跟着n 行输入。每组测试数据占1 行,包括一个正整
数b(1 <= b <= 92)
输出要求
n 行,每行输出对应一个输入。输出应是一个正整数,是对应于b 的皇后串
输入样例
2
1
92
输出样例
15863724
84136275

我的解题:根据题目的要求,我们可以先把这92种情况存储在一个数字里面。

  递归,每一行确定一个点之后,就可以把问题缩小为一个相同的,更小的问题。回溯,没确定一个点后,标记下来,当递归完成之后,再回复回来,继续找本行的下一个情况。

  中间还有判断是否符合规则,我这只判断同列、正反对角线这三种情况,因为每行只填一个数。

代码:

  1. #include <stdio.h>
  2. int queen[8][8];
  3. int nArr[93][8];
  4. int nCount;
  5. int Jude(int r,int c);
  6. void Queen(int row);
  7. void save();
  8. void print();
  9. int main()
  10. {
  11. int n,t;
  12. nCount = 0;
  13. Queen(0);
  14. //2printf("Count:%d\n",nCount);
  15. scanf("%d",&n);
  16. while(n)
  17. {
  18. scanf("%d",&t);
  19. print(t);
  20. n--;
  21. }
  22. return 0;
  23. }
  24. int Jude(int r,int c)
  25. {
  26. int i,j;
  27. //check col
  28. for(i = 0; i < r; i++)
  29. {
  30. if(queen[i][c] == 1)
  31. return 0;
  32. }
  33. //
  34. for(i = r-1,j = c-1; i >= 0 && j >= 0; i--,j--)
  35. {
  36. if(queen[i][j] == 1)
  37. return 0;
  38. }
  39. //
  40. for(i = r-1,j = c+1; i >= 0 && j <= 7; i--,j++)
  41. {
  42. if(queen[i][j] == 1)
  43. return 0;
  44. }
  45. return 1;
  46. }
  47. void Queen(int row)
  48. {
  49. int col;
  50. if (row == 8)
  51. {
  52. nCount++;
  53. save();
  54. }
  55. else
  56. {
  57. for(col = 0; col <= 7; col++)
  58. {
  59. if(Jude(row,col))
  60. {
  61. queen[row][col]=1;
  62. Queen(row+1);
  63. queen[row][col]=0;
  64. }
  65. }
  66. }
  67. }
  68. void save()
  69. {
  70. int row,col;
  71. for(row = 0; row <= 7; row++)
  72. {
  73. for(col = 0; col <= 7; col++)
  74. {
  75. if(queen[row][col] == 1)
  76. {
  77. nArr[nCount][row] = col+1;
  78. break;
  79. }
  80. }
  81. }
  82. }
  83. void print(int n)
  84. {
  85. int i;
  86. for(i = 0; i <= 7; i++)
  87. {
  88. printf("%d",nArr[n][i]);
  89. }
  90. printf("\n");
  91. }
  92. 2013/5/18 18:53

感觉经历了这次实习之后,我在代码规范性上有了很大的提高。这也得益于看别人的规范代码,多学习。

  我的思路是比较笨的方法。因为每次判断是否可以放置皇后的时候,都去遍历一遍和它相同的并在它前面的列、两个对角线的所有点,看它们有没有被放置。其实更好的办法是用三个数组来存储列和连个对角线的情况,因为只有8列,那么我们可以开个range数组来记录列,然后开个lineA[17]数组来记录135度对角线,同理lineB[17]数组记录45度对角线。因为每个对角线都有共性,比如135度对角线它们每个点的row+col是相等的,45度对角线它们的row-col+9同样是相同的。因此用三个数组来存储可以大大减少我们的代码量,非常好的方法!要学习!!

  1. #include <stdio.h>
  2. int queen[8][8];
  3. int nArr[93][8];
  4. int range[9],lineA[17],lineB[17];//col用来标示列,lineA用来标示135度对角线,lineB用来标示45度对角线
  5. int mark[9];
  6. int nCount;
  7. void Queen(int row);
  8. void print(int n);
  9. int main()
  10. {
  11. int n,t,i;
  12. nCount = 0;
  13. for(i = 0; i <= 8; i++)
  14. {
  15. range[i] = 1;
  16. }
  17. for(i = 0; i <= 16; i++)
  18. {
  19. lineA[i] = lineB[i] = 1;
  20. }
  21. Queen(0);
  22. //printf("Count:%d\n",nCount);
  23. scanf("%d",&n);
  24. while(n)
  25. {
  26. scanf("%d",&t);
  27. print(t);
  28. n--;
  29. }
  30. return 0;
  31. }
  32. void Queen(int row)
  33. {
  34. int col;
  35. if (row == 8)
  36. {
  37. nCount++;
  38. for(col = 0; col <=7; col++)
  39. {
  40. nArr[nCount][col]=mark[col];
  41. }
  42. }
  43. else
  44. {
  45. for(col = 0; col <= 7; col++)
  46. {
  47. if(range[col] && lineA[row-col+9] && lineB[row+col])
  48. {
  49. mark[row] = col;
  50. range[col]=lineA[row-col+9]=lineB[row+col] = 0;
  51. Queen(row+1);
  52. range[col]=lineA[row-col+9]=lineB[row+col] = 1;
  53. }
  54. }
  55. }
  56. }
  57. void print(int n)
  58. {
  59. int i;
  60. for(i = 0; i <= 7; i++)
  61. {
  62. printf("%d",nArr[n][i] + 1);
  63. }
  64. printf("\n");
  65. }
  66. 2013/5/19 11:57

  

转载于:https://www.cnblogs.com/Jason-Damon/archive/2013/05/18/3085862.html

发表评论

表情:
评论列表 (有 0 条评论,416人围观)

还没有评论,来说两句吧...

相关阅读

    相关 --皇后

      八皇后这个典的问题,是每个真正程序员必须经历过的。这也是我第二次来解决这个问题了,第一次应该是学数据结构那时候吧。这次写起来顺利多了,基本没遇到什么卡壳的地方。递归+回溯。