zoukankan      html  css  js  c++  java
  • 数据结构 —— N皇后

    你本来是有机会的,
    但是你输了,
    你不能活在过去。
    ——派大星《海绵宝宝》

    数据结构 —— N皇后


    C语言程序

    可以更改N值而修改棋盘规格

    /*******************************************
     *
     * 回溯法N皇后
     * create: 2020年6月4日19点59分
     * author: LOS
     *
     *******************************************
     * 此程序作法与回溯法的01背包类似
     *******************************************
     * 空间树不是二叉树,在回溯法中,空间树是存储了
     * 每一步所可能的所有结果,并且遍历该结果寻找到
     * 最优路径。
     *******************************************/


    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>

    #define MAX_CAPACITY 500 // 可操作数据的最大容量
    #define N 5 // 皇后的总数量

    typedef int Elem; // 定义最小元素

    typedef struct Node{
        struct Node *nodes[N]; // 子节点
        Elem curChessBoard[N][N]; // 存储棋盘的状态 0表示没有皇后,1表示有皇后
    } Node;

    struct {
        Node Nodes[MAX_CAPACITY];
        int size;
    } NodeCapacity; // 存储所有可操作数据的容器

    Node* rootNode;
    int count = 0;

    void init() // 初始化全局变量
        NodeCapacity.size = 0;
    }

    Node* createChessBoard() {
        Node* p = &NodeCapacity.Nodes[NodeCapacity.size];
        NodeCapacity.size++;
        return p;
    }

    void printChessBoard(Elem chessBoard[N][N]) {
        int j, k;
        count++;
        printf("第%d种情况: ", count);
        printf("----------------- ");
        for (j = 0; j < N; j++) { // 行
            for (k = 0; k < N; k++) {
                printf("%d ", chessBoard[j][k]);
            }
            putchar(' ');
        }
        printf("----------------- ");
    }

    void trackBack(Node *node,int layer) // layer层级
        int i,j,k;
        int disabledLines[N] = { 0 };
        if (layer == N) {
            printChessBoard(node->curChessBoard); // 输出棋盘
            return// 到最终子叶节点则直接出栈
        }

        // 判断剪枝  跳过根
        for (i = 0; i < N; i++) { // 指定某一行
            for (j = 0; j < layer; j++) { // 针对i行
                if (node->curChessBoard[i][j]) {
                    disabledLines[i] = 1;
                }
            }
            for (j = 0; j < N; j++) { // 针对列(此为不可能存在的现象,不作判断)
                // 此处留着更改遍历策略时使用
            }
            for (j = i + 1, k = layer - 1; j < N && k >= 0; j++, k--) { // 左下方向寻找
                if (node->curChessBoard[j][k]) {
                    disabledLines[i] = 1;
                }
            }
            for (j = i - 1, k = layer - 1; j >= 0 && k >= 0; j--, k--) { // 左上方向寻找
                if (node->curChessBoard[j][k]) {
                    disabledLines[i] = 1;
                }
            }
            for (j = i - 1, k = layer + 1; j >= 0 && k < N; j--, k++) { // 右上方向寻找(同理不作判断)
                // 此处留着更改遍历策略时使用
            }
            for (j = i + 1, k = layer + 1; j < N && k < N; j++, k++) { // 右下方向寻找(同理不作判断)
                // 此处留着更改遍历策略时使用
            }
        }

        for (i = 0; i < N; i++) { // 在行上放皇后
            if ( disabledLines[i] ) continue// 被剪枝的部分跳过
            node->nodes[i] = createChessBoard(); // 新建一个节点
            memcpy(node->nodes[i]->curChessBoard, node->curChessBoard, sizeof(node->curChessBoard)); // 复制数组
            node->nodes[i]->curChessBoard[i][layer] = 1//放置皇后
            trackBack(node->nodes[i], layer + 1);
        }

    }

    void main() {
        rootNode = createChessBoard(); // 有根节点即可
        memset(rootNode->curChessBoard, 0sizeof(rootNode->curChessBoard));
        trackBack(rootNode, 0);
    }

    运行结果

    第1种情况:
    -----------------
    1 0 0 0 0
    0 0 0 1 0
    0 1 0 0 0
    0 0 0 0 1
    0 0 1 0 0
    -----------------
    第2种情况:
    -----------------
    1 0 0 0 0
    0 0 1 0 0
    0 0 0 0 1
    0 1 0 0 0
    0 0 0 1 0
    -----------------
    第3种情况:
    -----------------
    0 0 1 0 0
    1 0 0 0 0
    0 0 0 1 0
    0 1 0 0 0
    0 0 0 0 1
    -----------------
    第4种情况:
    -----------------
    0 0 0 1 0
    1 0 0 0 0
    0 0 1 0 0
    0 0 0 0 1
    0 1 0 0 0
    -----------------
    第5种情况:
    -----------------
    0 1 0 0 0
    0 0 0 1 0
    1 0 0 0 0
    0 0 1 0 0
    0 0 0 0 1
    -----------------
    第6种情况:
    -----------------
    0 0 0 0 1
    0 0 1 0 0
    1 0 0 0 0
    0 0 0 1 0
    0 1 0 0 0
    -----------------
    第7种情况:
    -----------------
    0 1 0 0 0
    0 0 0 0 1
    0 0 1 0 0
    1 0 0 0 0
    0 0 0 1 0
    -----------------
    第8种情况:
    -----------------
    0 0 0 0 1
    0 1 0 0 0
    0 0 0 1 0
    1 0 0 0 0
    0 0 1 0 0
    -----------------
    第9种情况:
    -----------------
    0 0 0 1 0
    0 1 0 0 0
    0 0 0 0 1
    0 0 1 0 0
    1 0 0 0 0
    -----------------
    第10种情况:
    -----------------
    0 0 1 0 0
    0 0 0 0 1
    0 1 0 0 0
    0 0 0 1 0
    1 0 0 0 0
    -----------------
  • 相关阅读:
    win8.1下解决Visual C++不兼容的方法
    Java文件File类学习总结
    Java可视化日历(Date类、DATe Format类、Calendar类综合运用),开发可视化日历小程序
    java时间日期类(Date、DateFormat、Calendar)学习
    使用SimpleDateFormat类来实现时间跟字符串的转化
    Java中运行时异常和非运行时异常什么鬼?
    Java构造器练习题
    Word中一条删除不掉的单或双横线的解决办法
    Java中常用类(包装类扩展知识)
    Java异常学习总结二
  • 原文地址:https://www.cnblogs.com/rcklos/p/13046666.html
Copyright © 2011-2022 走看看