走迷宫!

自动化走迷宫

做maze题时不用自己在看花眼的maze里找路线了!!!

1.策略


  为了能回溯,也就是没路走时可以往回走, 这里采用了栈存储之前的路线,maze则采用了vector二位数组存储。
  那么简单地想,每一点位对四个方向进行检测,不断走下去,直到没路可走,便出栈往回走。这样会碰到什么问题呢? 由于进入一个点位的方向未知, 在试探四个方向时,可能会走回到上一个点位, 那就把该方向屏蔽,这样就需要给每个方向偏移打上一个标记位, 区分是否会回到上一个点位。
  我们采用另一种办法, 即将走过的点位标记为障碍物, 从而避免了自发地走回去的可能性, 而当没路走时, 仍然可以通过栈回溯到之前的点位。

  具体细节请看代码

2.code

先来栈代码
偏移结构和坐标结构也定义在这里,因为这个数据结构是按c风格写的,所以不太方便,只能把结构体定义在这里。

#include <stdio.h>
#include <malloc.h>
#include <stdbool.h>


typedef struct coordinate ElementType;
typedef struct SNode* PtrToSNode;


//坐标
struct coordinate
{
    int row;
    int col;
};

//行动偏移, row表示行偏移, col表示列偏移
struct offsets
{
    int row;
    int col;
};

//初始定义
typedef struct SNode
{
    ElementType* Data;
    int Top;
    int MaxSize;
}* Stack;

//创建一个栈空间
Stack CreateStack(int MaxSize)
{
    Stack s = (Stack)malloc(sizeof(struct SNode));
    s->Data = (ElementType*)malloc(MaxSize * sizeof(ElementType));
    s->Top = -1;
    s->MaxSize = MaxSize;
    return s;
}

//判断栈是否满
bool Isfull(Stack s)
{
    return s->Top == s->MaxSize - 1;
}

//判断栈是否空
bool IsEmpty(Stack s)
{
    return s->Top == -1;
}

//压栈
bool Push(Stack s, ElementType element)
{
    if(Isfull(s))
    {
        printf("Stack is full!");
        return false;
    }
    else 
    {
        s->Data[++(s->Top)] = element;
    }

    return true;
}

//出栈
ElementType Pop(Stack s)
{
    if(IsEmpty(s))
    {
        printf("Stack is empty!");
        return {0, 0};
    }
    else
    {
        return s->Data[s->Top--];
    }
}

然后是主要的实现代码。

//走迷宫,哈哈

#include "stack.h"
#include <iostream>
#include <vector>

using namespace std;

int main()
{
    cout << "请分别输入可行块、障碍块、入口、出口" << endl;
    char able, hinder, entry, exit;
    cin >> able >> hinder >> entry >> exit;
    //分别代表四种方块

    getchar();                                                                  //读取这里的换行符

    Stack s = CreateStack(0x80);                                                //创建一个栈
    struct offsets mov[4] =                                                     
        {{-1, 0}, {1, 0}, {0, -1}, {0, 1}};                                     //偏移数组,0-3四个下标对应上下左右偏移
    vector<char> Row1{hinder};                                                  //新建一个vector容器用来储存第一行,以便于判断列数
                                                                                //在第一列加了边界
    char input;                                                                 //输入缓冲

    cout << "请输入迷宫矩阵, 以“!”表示迷宫输入结束。" << endl;

    while ('\n' != (input = getchar()))                                         //一直读到换行符
    {
        Row1.push_back(input);
    }

    Row1.push_back(hinder);
    int column_size = Row1.size();                                              //在末尾加边界,并计算长度

    vector<vector<char>> maze{vector<char>(column_size, hinder), Row1};         //创建二位的数组, 初始化了前两行,第一行是边界

    while (true)
    {
        cin >> input;
        if (input == '!')
            break;
        maze.push_back(vector<char>{hinder, input});
        int num_row = maze.size() - 1;
        for (int i = 2; i <= column_size - 2; i++)
        {
            cin >> input;
            maze[num_row].push_back(input);
        }
        maze[num_row].push_back(hinder);
    }                                                                           //重复读取直到!, 这里内层循环便于在首尾加边界

    maze.push_back(vector<char>(column_size, hinder));
    int row_size = maze.size();                                                 //在最后加边界, 计算行数

    struct coordinate cd_entry;
    for (int i = 0; i < row_size; i++)
    {
        for (int j = 0; j < column_size; j++)
        {
            if (maze[i][j] == entry)
            {
                cd_entry = {i, j};
                break;
            }
        }
    }                                                                           //检索, 找到入口点

    /* 至此,初始化完毕, 开始走迷宫 */

    struct coordinate next_cd, now_cd;                                          //定义现在坐标、下一步坐标
    int direction = 0, found = 0;                                               //方向(偏移数组的下标), 是否找到出口的标记
    vector<int> route{0};                                                       //储存路线的vector容器
    Push(s, cd_entry);                                                          //初始压栈, 便于待会给定入口坐标
    maze[cd_entry.row][cd_entry.col] = hinder;                                  
    //以下的策略是进入一个点,就将其标记为障碍物,从而杜绝了自发走回已过点位的可能性(靠出栈能往回走), 这里将入口点标记为障碍物

    while (!IsEmpty(s) && !found)                                               //栈空时表示前方无路,通过出栈一直回到了起点
    {
        now_cd = Pop(s);                                                        //回到上一个点位,四个方向都不行时会回到这里。
        route.pop_back();                                                       //删除路线中刚刚记录的行动
        direction = 0;                                                          //重置方向

        //出发
        while (direction < 4 && !found)
        {
            next_cd = {now_cd.row + mov[direction].row, now_cd.col + mov[direction].col};
            if (maze[next_cd.row][next_cd.col] == exit)                         //如果是出口
            {
                route.push_back(direction);
                found = 1;
            }
            else if (maze[next_cd.row][next_cd.col] == able)                    //如果下一个点可以走(没走过且不是障碍物)
            {
                route.push_back(direction);
                Push(s, now_cd);
                maze[next_cd.row][next_cd.col] = hinder;
                now_cd = next_cd;
                direction = 0;
            }
            else
                direction++;                                                    //换一个方向
        }
    }

    cout << endl;
    if (found)
    {
        //终于出来了!呼...呼...呼.........
        for (int i = 0; i < route.size(); i++)
        {
            switch (route[i])
            {
            case 0:
                cout << 'w';
                break;
            case 1:
                cout << 's';
                break;
            case 2:
                cout << 'a';
                break;
            case 3:
                cout << 'd';
                break;
            }
        }
    }
    else
        cout << "被困死了!!!" << endl;         //寄

    return 0;
}

3.改进

  学完栈的实现之后临时起意写的maze程序,有诸多问题,如stack是c风格,不清楚怎么在cpp文件中定义其元素, 所以最后是在h文件中定义了元素结构体。如果改成c++风格的stack,也就是使用模板类写这个stack,会好很多, 不过stl中其实直接就有stack的模板,直接调用也可以, 用模板的方便之处在于,可以直接创建存储不同元素的stack了, 而且函数都是封装好的,比每次还要传入栈这个参数方便。
  再者,看过其他实现方式,也可以定义另外一个矩阵,以存储走过的点位,就不用每次走到某个点位把他变成hinder了(这可能更不方便了)
  然后,这个程序要求最后输入一个!以提示输入完成, 如果在输入时直接指明行数和列数,代码会简单一些,我想尽可能简化使用程序的人的操作,也就是尽可能更智能,可以直接按输入的情况存储矩阵, 但又没想到更好的解决办法,便这么干了。




  还有诸如代码风格等可能存在的其他问题,希望能得到大佬的指正。

posted @ 2022-05-24 23:24  点墨留白  阅读(75)  评论(0编辑  收藏  举报