八皇后的C语言实现

唯一的一门语言课Java课上老师讲过这个问题,使用“回退”的方法,期末考试也考过,但是这么多年来一直没有亲自去做过它,即使偶尔想到也觉得太过麻烦,没有那个能力实现。但是前两天一个兄弟作出来了,我觉得我也应该作出来,要不然实在没面子,于是使用C语言来实现一下。
一开始,我使用struct(A)来保存每一个皇后的位置,另外使用了一个struct(B)来保存当前列之前所有列中已放过皇后的位置,以防重复处理;我觉得方法思想上是没有问题的,但是运行的时候一个结果都没有出来,进入了死循环。我发现是往B中插入的数据没有成功,虽然在那个小funcion里边是没问题了,出来之后就回到原样了,因此已放过皇后的位置总是没有保存,每次都回退到之前的位置。
后来,我想到了另外一种实现,不使用struct,使用两个数组,分别保存每一列皇后的位置,已经皇后走过的标志(使用了2的row次方的方法,实际上跟前一个数组差不多),然后按照这个思路的结果,前边4个答案出来了,然后死循环,直到stack overflow。当我仔细分析算法的时候,我发现另一种更简洁一点的实现,只用一个数组,因为每一个数组元素储存的就是该列已经尝试过的最大值,下一次应该从它的下一个位置寻找。按照这个思路实现的结果跟上一个一样,只出来了4个答案(还差了88个)。
我再分析,发现是第一列第一个位置的所有可能情况找出来之后没法跳出循环,寻找第一列第二个位置的情况,于是我添加了
    if (i>=ROW && column==1)
        break;         /*45-46行*/   
如果第二列已经找到头了,就跳出。这个结果是得到了88个,看来最后一个位置(第一列)还有问题。于是我再添加了
    else if (column==0 && i>5)
        break;        /*47-48行*/
当回退到第一列的时候,第二列如果序号超过了5,就跳出,因为序号6和7是肯定无法匹配的。这样就没有问题了。
但是这样修改我觉得不够清晰明了,难解释,今早我想除了另外一个办法,在之后某个地方添加
    if (coulmn == 0)
        break;        /*73-74行*/
这是回退的时候如果退到第一列了,说明第一列当前位置已经搜索完毕,可以进入下一个位置了,根据实现,这里的循环就应该出来了。
具体的实现参看如下的源代码:

  1. /**
  2.  * Author: Hegc Huang
  3.  * Copyright 2008
  4.  */
  5. #include <stdio.h>
  6. #include <memory.h>

  7. #define ROW      8
  8. #define COLUMN   ROW

  9. int success = 0;    /*successful times*/

  10. void show(int* p)
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
二、 算法思想: 采用回溯法解决八皇后问题。从第一行开始,放第一个皇后,放好皇后以后,她所在的行,列和对角线上的每一个位置就是她的管辖范围,别的皇后没有权利干涉,否则死无藏身之地。 然后,第二个皇后,从第二行的第一列开始判断所在的位置是否是别的皇后的管辖范围,找到第一个还没有被占据的位置,则将其占为己有。暂时,该皇后停在该位置。然后,第三个到第八个皇后依次从第三行,第四行,… ,到第八行的第一列开始寻求自己的位置。假如到第i个皇后时,已经没有任何位置可选,则第i-1个皇后必须往后移动进行协调,同样,假如第i-1个皇后往后移动时没有找到空位置,则第i-2个皇后必须往后移动,进行协调,当找到空位置时,暂时停下,将下一个皇后重新从第一列开始寻找空位置。重复上述过程,直到所有皇后都停下来。则得到了第一个解。要想产生所有的解,则当产生第一个解以后,第八个皇后往后移动,找下一个可以利用的空位置,找不到,则第七个皇后必须往后移动,若找到空位置则停下,第八个皇后从第八行第一列重新试探,找到空位置。一直这样,直到第一个皇后将第一行遍历完。得到的解就是所有解。 三、 概要设计: ***************类型及相关变量定义***************** //位置信息类型 typedef struct { int row; int col; }PosType; //皇后类型 typedef struct Queen{ PosType pos; int number; //第几号皇后 }QueenType; //栈节点类型 typedef struct Note{ QueenType queen; struct Note *next; }NoteType; //棋盘,某一位置chessboard[i][j]上有皇后,则该位的值变为皇后序号。同样,该皇后的势 //力范围内的位置上的值全部变为该皇后的序号。 int chessboard[8][8]; //结果集,共92种解,每一种解中记录8个位置信息。 PosType ResultSet[92][8]; //定义一个栈,保存信息 Typedef struct{ NoteType head; Int size; }QueenStack; //定义一个栈,存放皇后信息 QueenStack qstack; *************相关操作**************** //初始化棋盘,开始时每个位置上都没有皇后,值全为0;并给8个皇后编号。 void initChessboard(); //回溯求八皇后问题的所有解,皇后协调算法 void queenCoordinate(); //输出所有解 void printResult();
皇后问题是指在8×8的国际象棋棋盘上放置8个皇后,使得任意两个皇后都不能在同一行、同一列或同一条对角线上,问有多少种摆法。下面是C语言实现皇后问题的代码示例: ```c #include <stdio.h> #include <stdlib.h> #define N 8 int count=0; //记录符合条件的方案数 //输出解法 void print_solution(int chessboard[N][N]) { int i, j; printf("Solution %d:\n", count+1); for (i = 0; i < N; i++) { for (j = 0; j < N; j++) { if (chessboard[i][j] == 1) { printf("Q "); } else { printf(". "); } } printf("\n"); } printf("\n"); } //判断该位置是否可以放置皇后 int is_valid(int chessboard[N][N], int row, int col) { int i, j; //判断同一列是否有皇后 for (i = 0; i < row; i++) { if (chessboard[i][col] == 1) { return 0; } } //判断左上方是否有皇后 for (i = row-1,j = col-1; i >= 0 && j >= 0; i--,j--) { if (chessboard[i][j] == 1) { return 0; } } //判断右上方是否有皇后 for (i = row-1,j = col+1; i >= 0 && j < N; i--,j++) { if (chessboard[i][j] == 1) { return 0; } } return 1; } //递归求解八皇后问题 void eight_queens(int chessboard[N][N], int row) { int i, j; //已经放置了8个皇后,输出解法 if (row == N) { print_solution(chessboard); count++; return; } //尝试在该行的每一个位置放置皇后 for (i = 0; i < N; i++) { if (is_valid(chessboard, row, i)) { chessboard[row][i] = 1; eight_queens(chessboard, row+1); chessboard[row][i] = 0; } } } int main() { int chessboard[N][N] = {0}; //初始棋盘为空 eight_queens(chessboard, 0); if (count == 0) { printf("No solution found.\n"); } else { printf("Total number of solutions: %d\n", count); } return 0; } ``` 代码思路: 1.定义一个8×8的数组作为棋盘,初始时所有元素都为0,表示没有皇后。 2.定义一个计数器count,用于统计符合条件的方案数。 3.定义一个函数is_valid,用于判断当前位置是否可以放置皇后。如果同一列、左上方或右上方已经有皇后,则无法再在该位置放置皇后。 4.定义一个函数print_solution,用于输出符合条件的解法。 5.定义一个递归函数eight_queens,用于求解八皇后问题。该函数的参数chessboard表示当前棋盘的状态,参数row表示当前正在尝试在第几行放置皇后。 6.递归求解的思路是从第0行开始,尝试在该行的每一个位置放置皇后,如果该位置可以放置皇后,则继续递归求解下一行。如果递归到第8行,说明已经成功放置了8个皇后,输出解法并增加计数器count。 7.最后在main函数中调用eight_queens函数求解八皇后问题,如果count为0表示没有符合条件的解法。 希望可以帮到您!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值