深度优先算法和Ncurses - 迷宫

前言:

阅读此篇文章要求读者具有一些基础的ncurses库了解

深度优先算法是将各种情况分配一定的优先级,然后在某个问题出现的时候,选择优先级高的情况进行处理,特定情况下进行回退,选择优先级次之的情况进行尝试. 点击这里可以得到源码

我这里没有考虑没有出口的情况下,深度优先算法的行为结果。之后会有专门的研究文章。


程序主体思路

过程

  • 创建一个迷宫
  • 创建一个人
  • 把人放到迷宫的入口
  • 深度优先算法控制人,走到出口。
  • 退出

代码

#include <ncurses.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>


// this is the myth map.
char myth[][31] = {

    "******************************", 
    "              *          *   *",
    "*  ****       *        *    ->",
    "*     *       * ******   *****",
    "********      *              *",
    "*   **    *   *  * ******** **",
    "* * *       *    *           *",
    "* * ******* *    *  ******** *",
    "* *         *    *           *",
    "* * *  *    *    *    ***    *",        
    "******************************"
};

这里用一个字符数组来存储迷宫的信息

char book[12][30] = {0};

book数组负责这个点有没有走过的标记

int myth_starty = 0;
int myth_startx = 0;

myth_starty, myth_startx 分别负责在myth WINDOW里面的迷宫左上角起点坐标的y和x, 因为在ncurses里面,的坐标和我们的迷宫数组下标之间需要一个翻译。

// this is a person struct.
typedef struct _person
{
        int x;
        int y;
    int step;

}Person;

这是一个结构体,来存放player的信息

// this a struct for a point in game.
typedef struct _loc
{
    int x, y;
}MLoc;

一个包装了y和x两个int值的结构体, 没怎么用

// this function handle the person's moving.
void moveTo(WINDOW *w, Person *p, int y, int x);

这个函数负责把一个Person移动到某个窗口w的y行x列

// this function return a person pointer.
Person* con_person();

这个函数负责construct一个person,返回Person指针

// this function destroy a person.
void destroy_person(Person *p);

这个函数释放Person指针p指向的堆内存

// this funcion creates a myth in the middle of the target window.
int create_myth(WINDOW *target);

这个函数负责打印迷宫到窗口target的中心

//
void go_through(WINDOW *, Person *);

这个函数就是负责让小人走出迷宫

int main()
{
    // preparation for this program.
    initscr();
    clear();
    curs_set(0); // hide the cursor
    noecho(); // no echo.
    cbreak(); // close the line buffer.

ncurses程序初始化准备部分

init_pair(1, COLOR_RED, COLOR_BLACK);
init_pair(2, COLOR_BLACK, COLOR_WHITE);

颜色效果不太好,然后就取消掉了。这两行代码无用

// make a border for the standred screen.
wborder(stdscr,  '|', '|', '-', '-', '*', '*', '*', '*');
refresh();

给标准屏幕添加一个边框

// two var for the max size of the stdscr.
int scry, scrx;
getmaxyx(stdscr, scry, scrx);

// 把标准屏幕的宽度和高度存放在scrx,scry里面

// use a LOC var for an area. y stands for height, x stands for width.
MLoc title_part;
title_part.y =     (scry-15)/2 + 4;
title_part.x =  (scrx-50)/2;    

// 计算出title_part的大小, 用来存放一些程序信息

/* print some game info */
char *title = " GAME: Crack the myth ";
char *author = " AUTHOR: Jack ";
char *version = " VERSION: 1.0.1 ";



attron(A_REVERSE);
mvprintw((title_part.y - 3)/2, (scrx-strlen(title))/2, "%s", title);
attroff(A_REVERSE);    

attron(A_UNDERLINE);
mvprintw((title_part.y - 3)/2 + 1, (scrx-strlen(version))/2, "%s", version);
mvprintw((title_part.y - 3)/2 + 2, (scrx-strlen(author))/2, "%s", author);
attroff(A_UNDERLINE);

refresh();

把名称、作者信息、版本号打印在游戏界面中

// myth is a window for the myth map.
WINDOW *w_myth = newwin(15, 50, (scry-15)/2 + 4, (scrx-50)/2 - 20);

// we called the function which we defined to show the myth map
create_myth(w_myth);
wborder(w_myth,  '|', '|', '-', '-', '+', '+', '+', '+');
// 
mvwprintw(stdscr,  (scry-15)/2 + 4 + 6, (scrx-50)/2 - 20 + 60, "Cracking...");
refresh();
wrefresh(w_myth);

创建一个窗口来存放迷宫地图,并且给其添加边界

// put a person on myth.
Person *player = con_person();
moveTo(w_myth, player, 1, 0);


getchar();

随便得到一个字符之后,开始进行寻路

    go_through(w_myth, player);


    // get a character then exit.
    getchar();
    endwin();

    return 0;
}

以上就是程序的全部流程,不难 对吗?

深度优先算法

过程

我们测试每一个点,如果当前位置在终点位置上,那么我们就直接结束程序。或者我们就来计算下一个点的坐标。下一个点的坐标可以是右边的、下边的、左边的、上边的、(东南西北),我们在这里用一个next二维数组来处理,如果是下一个点是邻接的右边的点,那么y坐标不变,x坐标+1, 所以在next数组里面对应的是{0, 1} ( {y, x} ) 这样。

在后面用for循环遍历每一种情况的时候,会用到这个next数组。

代码

void go_through(WINDOW *w, Person *p)
{

    int cy, cx;
    int ny, nx;

    if ( p->y == 2 && p->x == 29) 
    {
        attron(A_REVERSE);
        mvwprintw(stdscr,  (LINES-15)/2 + 4 + 6, (COLS-50)/2 - 20 + 60, "    DONE   ");
        attroff(A_REVERSE);
        wrefresh(stdscr);
        wrefresh(stdscr);
        getchar();
        endwin();
        exit(0);
    }
    int next[4][2] = 
        {
            {0, 1},
            {1, 0},
            {0, -1},
            {-1, 0},
        };



    for (int i = 0; i < 4; ++i) 
    {
        // 计算下一个点的坐标
        ny = p->y + next[i][0]; 
        nx = p->x + next[i][1];

        // 保存当前坐标
        cy = p->y;
        cx = p->x;

        // 如果下个坐标值非法(越界),那么就跳过这个尝试
        // 在这里,每个循环代表的是每个方向上的尝试
        if (ny > 10 || ny < 0 || nx > 29 || nx < 0)
        {
            continue;
        }

        // 如果下一个坐标没有走过,并且不是墙
        if (book[ny][nx] == 0 && myth[ny][nx] != '*')
        {
            // 标记走过了这个点
            book[ny][nx] = 1;
            // 移动Person p走到这个点
            moveTo(w, p, ny, nx);
            // 睡100毫秒
            usleep(100000);
            // 以新的点作为起点进行尝试, 这就是递归算法。
            go_through(w, p);
            // 尝试结束后(可能是尝试失败的情况下), 退回来。 在for循环的下一次循环中尝试其他方向
            moveTo(w, p, cy, cx);
            // book[ny][nx] = 0;
        }
    }

    return ;

}

总结

学习算法总是: 纸上得来终觉浅, 绝知此事要躬行。 另外,这个程序只是寻找一条路,而不是寻找最短路径。

posted @ 2019-03-16 10:54  不怕旅途多坎坷  阅读(188)  评论(0编辑  收藏  举报