迷宫问题(规范格式且有大量注释)

迷宫问题(随机路径生成+回溯求解迷宫通路+可视化)

一、设计要求

1.1问题描述

(1) 迷宫问题描述:是一个经典的路径生成和寻路问题,假设有一个n×n矩形网格迷宫,其中一些单元格是由特殊符号形成的障碍物或者墙体而无法通行(障碍物以符号“1”和”-1”进行标记)。初始给定一个起点和终点,然后根据回溯算法找出一条由起点到终点的坐标路径。移动的方法只能是上、下、左、右四个方向,不能斜着移动,不能穿过墙壁。
(2) 随机prim算法为雏形的路径生成算法:将迷宫看成是一棵树,然后在初始化时候设置一些类似与结点的标记,然后通过以prim算法思想为基础生成所需要的树,生成时候选择结点连接的方式将觉得树的形状,最终得到一个无环的迷宫。简单迷宫类似于深度随机,复杂迷宫类似于完全随机抽取一个结点进行连接。
(3) 迷宫各项功能衔接的数据形式:编码-转码-解码,编码主要用于各种逻辑运算和算法实现的操作,转码用于统一储存迷宫的数据形式,解码用于最终可视化迷宫。
(4) 回溯算法:回溯算法是一种基于深度优先搜索的算法,通常用于在一个大状态空间中寻找一个可行解的问题。在问题求解时,将大问题划分为一个个小的子问题进行解决,其复杂度往往非常高,因为它是穷举遍历,其算法的关键在于如何剪枝以减少搜索的空间和保持搜索沿着正确的方向进行。

1.2需求分析

(1) 设计产生式系统:

  1. 综合数据库:Sij = {0,1} (i
  2. 规则库:定义i0j0为当前所在迷宫的最新位置的行和列,即Si0j0=‘0’。
    a. 上移规则:if i0 - 1 >= 0 then i0 -= 1
    b. 右移规则:if j0 + 1 < n then j0 += 1
    c. 左移规则:if j0 - 1 >= 0 then j0 -= 1
    d. 下移规则:if i0 + 1 < n then i0 += 1
  3. 回溯策略:当目前位置没有规则可行,并且未能到达终点,则回溯一步。
    (2) 程序功能:通过随机生成一个迷宫,然后通过回溯算法找出一条通路。
    (3) 输入形式:迷宫大小应为n×n矩阵,n为大于4的奇数整型。
    (4) 输出形式:以可视化路径和坐标形式呈现路径反馈给用户。
    (5) 测试数据:输入不是规定的数据域时候会提供相应的提示和警告或重新输入。

二、概要设计

2.1存储结构设计

(1) 通过矩阵储存迷宫的布局设计,先用数字储存,然后转码为字符型,最后解码为迷宫形状。
(2) 迷宫的起点和终点坐标均用数字存放,迷宫维度数采用整型存放。
(3) 迷宫中行走的放行通过数组存放。
(4) 通过一个vector容器存放迷宫通路的坐标。

2.2系统功能设计

(1) 电脑生成迷宫。电脑生成迷宫模块能实现生成简单和复杂迷宫两种,并且获得迷宫的通路答案,可视化迷宫通路并且打印迷宫坐标。
(2) 人工生成迷宫。人工生成迷宫模块能手动按照给定的模板输入迷宫,然后获得迷宫通路答案,可视化迷宫通路并且打印迷宫坐标。

3.2 系统子程序及功能设计
(1) Status resetMazeData(); //重置迷宫程序的数据结构
(2) Status initMazeStro(); //初始化生成迷宫储存空间
(3) Status mazeTransformer(); //迷宫编码转化器
(4) Status mazeTransformerDisplay(); //打印迷宫形状
(5) Status mazeDisplay(); //迷宫通路可视化
(6) Status solveMaze_Maker(int site_x,int site_y); //迷宫问题解决方案生成器
(7) Status SMMR(int x,int y,int dir_flag); //递归挖墙开路
(8) Status simpleMaze_Menu(); //电脑简单迷宫展示界面
(9) Status simpleMaze_Maker(); //电脑简单迷宫生成器
(10) Status CMMR(int x,int y); //随机prim递归挖墙开路
(11) Status complexMaze_Menu(); //电脑复杂迷宫展示界面
(12) Status complexMaze_Maker(); //电脑复杂迷宫生成器
(13) Status computerMazeInti(int **tmp_dir,int tmp_borderS,int tmp_borderE,int (&tmp_End)[2]); //电脑生成迷宫的起步初始化
(14) Status computerMazeEnd(int tmp_borderE,int (&tmp_End)[2]); //电脑生成迷宫的结尾终点的连接处理
(15) Status computerMenu(); //电脑生成迷宫模块子菜单界面
(16) Status artiMaze_Menu(); //人工迷宫展示界面
(17) Status artiMaze_Maker(); //人工迷宫输入器+生成器
(18) Status artificicalMenu(); //人工生成迷宫子菜单界面
(19) int main() //主函数

四 详细设计

4.1 数据类型定义

(1) 宏定义
#define OK 1 //成功运行状态
#define ERROR 0 //失败运行状态
#define OVERFLOWS -1 //数据溢出
#define DENY 2 //当前状态无解
#define RESTART 3 //重置程序
#define Status int //运行状态
(2) 全局变量定义
int n; //迷宫维数
int **mazeInt; //手动输入迷宫样式
char **mazeChar; //迷宫符号转化
int **mazeFlag; //寻找通路时候标记迷宫格子是否被访问
int mazeStart[2]; //迷宫起点坐标
int mazeEnd[2]; //迷宫终点坐标
vector mazePassage; //存放通路坐标
vector tmp_complexmaze; //存放复杂迷宫生成的临时坐标
vector>tmp_simplemaze; //存放简单迷宫生成时候的临时标记坐标

五、完整代码

(ps:由于程序里面提示明确,就不展示使用过程了。)

    /************************************************************************\
作者信息:
    name:Yaung
    Email: [email protected]
版权声明:
    版权由Yaung所有,除教学外,未经允许不得拷贝本人作业
模块名称:
    迷宫问题系统模块
摘要:
    本模块是我的一个课程设计报告,实现两种以prim算法为雏形的自动
    生成简单与复杂的迷宫和手动输入迷宫还有回溯求解迷宫的功能。
其它说明:
    在codeblocks中默认UTF-8编码,在cmd中运行需要在main函数中
    设置编码形式。
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
    Yaung于2023年3月25日修改本模块,email: [email protected]
    修改原因:完善了部分界面和优化了算法复杂度
    Yaung于2023年4月18日修改本模块,email: [email protected]
    修改原因:增加了迷宫通路的可视化
    Yaung于2023年4月22日修改本模块,email: [email protected]
    修改原因:完善了手动输入迷宫形式错误的功能逻辑
    Yaung于2023年5月4日修改本模块,email: [email protected]
    修改原因:优化算法
\************************************************************************/
#include 
#include
#include 
#include
#include
#include
#include
#include
using namespace std;

#define OK 1        //成功运行状态
#define ERROR 0         //失败运行状态
#define OVERFLOWS -1         //数据溢出
#define DENY 2         //当前状态无解
#define RESTART 3       //重置程序
#define Status  int     //运行状态

//begin-数据结构声明
int n;  //迷宫维数
int **mazeInt = nullptr;  //手动输入迷宫样式
char **mazeChar = nullptr;    //迷宫符号转化
int **mazeFlag = nullptr; //寻找通路时候标记迷宫格子是否被访问
int mazeStart[2];   //迷宫起点坐标
int mazeEnd[2]; //迷宫终点坐标
vector> mazePassage;    //存放通路坐标
vector> tmp_complexmaze;    //存放复杂迷宫生成时候的临时标记坐标
vector>>tmp_simplemaze;     //存放简单迷宫生成时候的临时标记坐标
//end-数据结构声明

//begin-静态数据定义
const int dir[4][2]= {{-1,0},{0,1},{1,0},{0,-1}}; //行走的方向,上右下左
//end-静态数据定义

//begin-方法函数声明
Status resetMazeData();     //重置迷宫程序的数据结构
Status initMazeStro();     //初始化生成迷宫储存空间
Status mazeTransformer();       //迷宫编码转化器
Status mazeTransformerDisplay();    //打印迷宫形状
Status mazeDisplay();       //迷宫通路可视化
Status solveMaze_Maker(int site_x,int site_y);       //迷宫问题解决方案生成器

Status SMMR(int x,int y,int dir_flag);      //递归挖墙开路
Status simpleMaze_Menu();      //电脑简单迷宫展示界面
Status simpleMaze_Maker();      //电脑简单迷宫生成器

Status CMMR(int x,int y);       //随机prim递归挖墙开路
Status complexMaze_Menu();    //电脑复杂迷宫展示界面
Status complexMaze_Maker();    //电脑复杂迷宫生成器

Status computerMazeInti(int **tmp_dir,int tmp_borderS,int tmp_borderE,int (&tmp_End)[2]);  //电脑生成迷宫的起步初始化
Status computerMazeEnd(int tmp_borderE,int (&tmp_End)[2]);       //电脑生成迷宫的结尾终点的连接处理
Status computerMenu();         //电脑生成迷宫模块子菜单界面

Status artiMaze_Menu();       //人工迷宫展示界面
Status artiMaze_Maker();        //人工迷宫输入器+生成器
Status artificicalMenu();         //人工生成迷宫子菜单界面

int main();
//end-方法函数声明

/**
调试方法
**/
//void testInt()
//{
//    for(int i=0; i=n||site_x<0||site_y>=n||site_y<0)    //超出界限
    {
        return OVERFLOWS;
    }
    if((mazeInt[site_x][site_y]==1||mazeInt[site_x][site_y]==-1)||mazeFlag[site_x][site_y]==1) //是墙或者已经访问过了
    {
        return ERROR;
    }
    mazeFlag[site_x][site_y]=1;     //合理位置标记已经访问
    mazePassage.push_back({site_x,site_y});     //保存通路的坐标之一
    vector> rules;      //当前位置的规则集合,上右下左
    if(site_x-1>=0) //上
    {
        rules.push_back({dir[0][0],dir[0][1]});
    }
    if(site_y+1=0) //左
    {
        rules.push_back({dir[3][0],dir[3][1]});
    }
    while(!rules.empty())
    {
        int recallStatus = solveMaze_Maker(site_x+rules[rules.size()-1][0],site_y+rules[rules.size()-1][1]);    //后面的返回状态
        if(recallStatus==OK)    //找到通路
        {
            return OK;
        }
        else if(recallStatus==ERROR||recallStatus==OVERFLOWS)    //已经访问或者撞墙或者超出范围
        {
            rules.pop_back();
            continue;
        }
        else if(recallStatus==DENY) //当前位置无解
        {
            mazePassage.pop_back();
            rules.pop_back();
            continue;
        }
    }
    return DENY;//当前状态无解
}
/************************************************************************\
函数名称:
    SMMR
功能描述:
    通过递归生成迷宫道路
函数参数:
    预判位置左边x,y,前一个方向参数dir_flag
返回值:
    越界返回-1,成功求解返回1,无法访问返回0
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
    Yaung于2023年5月4日创建本模块,email: [email protected]
        修改原因:优化算法,由递归改为动态规划
\************************************************************************/
Status SMMR(int x,int y,int dir_flag)    //简单迷宫的递归创造通路,标记前一次的方向-1为无方向
{
    //存放方向往回导航,动态规划
    int tmp_g[n][n];
    for(int i =0; i>tmp_dir; //储存每个结点可选方向
    vectortmp_End;     //储存当前方向信息
    do
    {
        if(x<0||x>=n||y<0||y>=n)    //越界情况直接退出递归
        {
            continue;
        }
        if(x+2*dir[0][0]>0&&mazeInt[x+2*dir[0][0]][y+2*dir[0][1]]==5)   //上面格子的处理
        {
            mazeInt[x+2*dir[0][0]][y+2*dir[0][1]]=6;    //标记为已经分叉的结点
            tmp_dir.push_back({dir[0][0],dir[0][1],0});
        }
        if(y+2*dir[1][1]0&&mazeInt[x+2*dir[3][0]][y+2*dir[3][1]]==5)   //左边格子的处理
        {
            mazeInt[x+2*dir[3][0]][y+2*dir[3][1]]=6;
            tmp_dir.push_back({dir[3][0],dir[3][1],3});
        }

        if(tmp_dir.empty())     //当前结点没有可以继续拓展的路时候进行回退开分路
        {
            tmp_dir = tmp_simplemaze.back();    //压出保存的项目
            tmp_simplemaze.pop_back();

            int xx = x;
            x += 2*dir[(tmp_g[x][y]+2)%4][0];       //返回前一个结点
            y += 2*dir[(tmp_g[xx][y]+2)%4][1];
            dir_flag = tmp_g[x][y];
        }
        else        //当前结点有路可以探索采用随机方法
        {
            //存完后随机打乱容器中的坐标
            random_shuffle(tmp_dir.begin(),tmp_dir.end());
            //获取一个新位置
            tmp_End = tmp_dir.back();
            tmp_dir.pop_back();
            tmp_simplemaze.push_back(tmp_dir);  //保存还未访问的方向
            //当前采取方向
            x += tmp_End[0]+dir[tmp_End[2]][0];
            y += tmp_End[1]+dir[tmp_End[2]][1];
            if(mazeInt[x][y]==5||mazeInt[x][y]==6)    //是5表示未被访问过则常规打通
            {
                dir_flag = tmp_End[2];
                mazeInt[x+dir[(dir_flag+2)%4][0]][y+dir[(dir_flag+2)%4][1]] = 0;        //打通两者中间的墙
                tmp_g[x+dir[(dir_flag+2)%4][0]][y+dir[(dir_flag+2)%4][1]] = dir_flag;
                mazeInt[x][y] = 0;
                tmp_g[x][y] = dir_flag;
            }
            //用完就清除临时变量
            while(!tmp_dir.empty())
                tmp_dir.pop_back();
            while(!tmp_End.empty())
                tmp_End.pop_back();
            vector>().swap(tmp_dir);
            vector().swap(tmp_End);
        }

    }
    while(!tmp_simplemaze.empty()||!tmp_dir.empty());
//    for(int i = 0; i<4; ++i)
//    {
//        int k = rand()%4;
//        SMMR(x+dir[k][0]*2,y+dir[k][1]*2,k);  //步长应当为2
//    }
    return OK;
}
/************************************************************************\
函数名称:
    simpleMaze_Maker
功能描述:
    电脑简单迷宫生成器算法,生成只有一条路走到终点,十分简单
函数参数:
    无
返回值:
    执行成功:OK
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
\************************************************************************/
Status simpleMaze_Maker()      //电脑简单迷宫生成器
{
    /**
    随机生成迷宫数据结构
    **/
    resetMazeData();
    initMazeStro();    //初始化生成迷宫储存空间
    //随机生成起点和终点,0,1,2,3标记,分别代表上右下左边界上的起点或终端
//    int tmp_dir[4][2] = {{0,rand()%n},{rand()%n,n-1},{n-1,rand()%n},{rand()%n,0}};
    int **tmp_dir = new int*[4];
    for(int i =0; i<4; ++i)
    {
        if(i==0)
        {
            tmp_dir[i] = new int[2];
            tmp_dir[i][0] = 0;
            tmp_dir[i][1] = rand()%n;
        }
        else if(i==1)
        {
            tmp_dir[i] = new int[2];
            tmp_dir[i][0] = rand()%n;
            tmp_dir[i][1] = n-1;
        }
        else if(i==2)
        {
            tmp_dir[i] = new int[2];
            tmp_dir[i][0] = n-1;
//            do
//            {
            tmp_dir[i][1] = (tmp_dir[i-2][1]+n/2)%n;
//            }
//            while(abs(tmp_dir[i][1]-tmp_dir[i-2][1])>n)||(n<=4)||(n%2==0)||(n>=110))        //输入非整型错误
    {
        cout<<" * 输入迷宫维数应当为整型奇数(4<维度大小<110):";
        cin.clear(); // 清除输入流错误标记
        cin.sync();// 取走刚才输入流中的字符
    }
    do
    {
        simpleMaze_Maker();     //生成简单迷宫
    }
    while(mazeTransformer()==ERROR);

//    testInt();
    /**
    迷宫展示
    **/
    char choice;
    while(1)
    {
        system("cls");      //清屏
        cout<<"\n\t\t\t\t电脑简单迷宫展示界面\n"<>choice;
        while(choice!='1'&&choice!='2'&&choice!='3')        //判断选项是否为规定值
        {
            if(choice>'9'||choice<'0')      //输入类型错误提示
            {
                cout<<" * 警告!输入类型错误!!"<>choice;
            }
            else        //输入值错误提示
            {
                cout<<" * 请输入正确选项并按回车:";
                cin>>choice;
            }
        }
        switch(choice)      //进入电脑生成迷宫模块的功能函数
        {
        case '1':
            solveMaze_Maker(mazeStart[0],mazeStart[1]); //回溯生成通路
            cout<<"迷宫的通路:"<>choice;
        while(choice!='1'&&choice!='2'&&choice!='3')        //判断选项是否为规定值
        {
            if(choice>'9'||choice<'0')      //输入类型错误提示
            {
                cout<<" * 警告!输入类型错误!!"<>choice;
            }
            else        //输入值错误提示
            {
                cout<<" * 请输入正确选项并按回车:";
                cin>>choice;
            }
        }
        switch(choice)      //进入电脑生成迷宫模块的功能函数
        {
        case '1':
            resetMazeData();    //释放指针,重置程序
            return OK;
        case '2':
            resetMazeData();    //释放指针,重置程序
            return RESTART;
        case '3':
            resetMazeData();    //释放指针
            exit(0);
        }
    }
    return OK;
}
/************************************************************************\
函数名称:
    computerMazeInti
功能描述:
    电脑生成迷宫的起步初始化,默认上右为先
函数参数:
    无随机起点位置,起点边界,终点边界,临时终点位置
返回值:
    执行成功:OK
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
\************************************************************************/
Status computerMazeInti(int **tmp_dir,int tmp_borderS,int tmp_borderE,int (&tmp_End)[2])
{
    //起点
    mazeStart[0] = tmp_dir[tmp_borderS][0];
    mazeStart[1] = tmp_dir[tmp_borderS][1];
    mazeInt[mazeStart[0]][mazeStart[1]] = 2;    //2是标记地图中的起点
    //终点
    mazeEnd[0] = tmp_dir[tmp_borderE][0];
    mazeEnd[1] = tmp_dir[tmp_borderE][1];
    mazeInt[mazeEnd[0]][mazeEnd[1]] = 3;    //3是标记地图中的终点
    //边界初始化起步
    if(tmp_borderS==0)//如果为上边界起点
    {
        if(mazeInt[mazeStart[0]+dir[2][0]][mazeStart[1]+dir[2][1]]==5)//下面直接是 5
        {
            tmp_End[0] = mazeStart[0]+dir[2][0];    //临时终点的x
            tmp_End[1] = mazeStart[1]+dir[2][1];    //临时终点的y被标记好
            mazeInt[tmp_End[0]][tmp_End[1]] = 0;       //标记已经访问的地方
        }
        else
        {
            //斜边才是 5
            //以默认主上和右为准,越界相反即可
            if(mazeStart[1]+1=0)
            {
                tmp_End[0] = mazeStart[0]+dir[0][0];
                tmp_End[1] = mazeStart[1]+dir[0][1];
            }
            else
            {
                tmp_End[0] = mazeStart[0]+dir[2][0];
                tmp_End[1] = mazeStart[1]+dir[2][1];
            }
            //向一边打通墙
            mazeInt[tmp_End[0]][tmp_End[1]] = 0;       //标记已经访问的地方
            //确定起点
            tmp_End[0] = tmp_End[0]+dir[3][0];
            tmp_End[1] = tmp_End[1]+dir[3][1];
            mazeInt[tmp_End[0]][tmp_End[1]] = 0;       //标记已经访问的地方
        }
    }
    else if(tmp_borderS==2)//如果为下边界起点
    {
        if(mazeInt[mazeStart[0]+dir[0][0]][mazeStart[1]+dir[0][1]]==5)//上面直接是 5
        {
            tmp_End[0] = mazeStart[0]+dir[0][0];    //临时终点的x
            tmp_End[1] = mazeStart[1]+dir[0][1];    //临时终点的y被标记好
            mazeInt[tmp_End[0]][tmp_End[1]] = 0;       //标记已经访问的地方
        }
        else
        {
            //斜边才是 5
            //以默认主上和右为准,越界相反即可
            if(mazeStart[1]+1=0)
            {
                tmp_End[0] = mazeStart[0]+dir[0][0];
                tmp_End[1] = mazeStart[1]+dir[0][1];
            }
            else
            {
                tmp_End[0] = mazeStart[0]+dir[2][0];
                tmp_End[1] = mazeStart[1]+dir[2][1];
            }
            //向一边打通墙
            mazeInt[tmp_End[0]][tmp_End[1]] = 0;       //标记已经访问的地方
            //确定起点
            tmp_End[0] = tmp_End[0]+dir[1][0];
            tmp_End[1] = tmp_End[1]+dir[1][1];
            mazeInt[tmp_End[0]][tmp_End[1]] = 0;       //标记已经访问的地方
        }
    }
    return OK;
}
/************************************************************************\
函数名称:
    computerMazeEnd
功能描述:
    电脑生成迷宫的结尾处理,默认上右为先
函数参数:
    终点边界,临时终点位置
返回值:
    执行成功:OK
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
\************************************************************************/
Status computerMazeEnd(int tmp_borderE,int (&tmp_End)[2])
{
    //连接终点,以上和右为默认
    if(tmp_borderE==0)//如果为上边界起点
    {
        if(mazeInt[mazeEnd[0]+dir[2][0]][mazeEnd[1]+dir[2][1]]!=0)//下面不是0
        {
            //斜边才是0
            //以默认主上和右为准,越界相反即可
            if(mazeEnd[1]+1=0)
            {
                tmp_End[0] = mazeEnd[0]+dir[0][0];
                tmp_End[1] = mazeEnd[1]+dir[0][1];
            }
            else
            {
                tmp_End[0] = mazeEnd[0]+dir[2][0];
                tmp_End[1] = mazeEnd[1]+dir[2][1];
            }
            //向一边打通墙
            mazeInt[tmp_End[0]][tmp_End[1]] = 0;       //标记已经访问的地方
        }

    }
    else if(tmp_borderE==2)//如果为下边界起点
    {
        if(mazeInt[mazeEnd[0]+dir[0][0]][mazeEnd[1]+dir[0][1]]!=0)//上面不是0
        {
            //斜边才是 0
            //以默认主上和右为准,越界相反即可
            if(mazeEnd[1]+1=0)
            {
                tmp_End[0] = mazeEnd[0]+dir[0][0];
                tmp_End[1] = mazeEnd[1]+dir[0][1];
            }
            else
            {
                tmp_End[0] = mazeEnd[0]+dir[2][0];
                tmp_End[1] = mazeEnd[1]+dir[2][1];
            }
            //向一边打通墙
            mazeInt[tmp_End[0]][tmp_End[1]] = 0;       //标记已经访问的地方
        }
    }
    return OK;
}
/************************************************************************\
函数名称:
    CMMR
功能描述:
    复杂迷宫生成通路时候的随机prim递归算法
函数参数:
    当前位置x,y
返回值:
    执行成功:OK
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
\************************************************************************/
Status CMMR(int x,int y)
{
    vector tmp_End;     //临时终点位置
    do
    {
        //标记附近的合法位置为6
        if(x+dir[0][0]>0&&mazeInt[x+dir[0][0]][y+dir[0][1]]==1)   //上面格子的处理
        {
            mazeInt[x+dir[0][0]][y+dir[0][1]]=6;
            tmp_complexmaze.push_back({x+dir[0][0],y+dir[0][1],0});
        }
        if(y+dir[1][1]0&&mazeInt[x+dir[3][0]][y+dir[3][1]]==1)   //左边格子的处理
        {
            mazeInt[x+dir[3][0]][y+dir[3][1]]=6;
            tmp_complexmaze.push_back({x+dir[3][0],y+dir[3][1],3});
        }
        //存完后随机打乱容器中的坐标
        random_shuffle(tmp_complexmaze.begin(),tmp_complexmaze.end());
        //获取一个新位置
        tmp_End = tmp_complexmaze.back();
        tmp_complexmaze.pop_back();
        x = tmp_End[0]+dir[tmp_End[2]][0];
        y = tmp_End[1]+dir[tmp_End[2]][1];
        if(mazeInt[x][y]==5)    //是5表示未被访问过则常规打通
        {
            mazeInt[tmp_End[0]][tmp_End[1]] = 0;
            mazeInt[x][y] = 0;
        }
        else if(mazeInt[x][y]==0)   //如果是0证明已经被访问过,直接封了当前的路
        {
            mazeInt[tmp_End[0]][tmp_End[1]] = -1;
        }
        //用完就清除临时变量
        while(!tmp_End.empty())
            tmp_End.pop_back();
    }
    while(!tmp_complexmaze.empty());
    return OK;
}
/************************************************************************\
函数名称:
    complexMaze_Maker
功能描述:
    电脑复杂迷宫算法通过随机prim算法的雏形
函数参数:
    无
返回值:
    执行成功:OK
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
\************************************************************************/
Status complexMaze_Maker()      //电脑简单迷宫生成器
{
    /**
    随机生成迷宫数据结构
    **/
    resetMazeData();
    initMazeStro();    //初始化生成迷宫储存空间
    //随机生成起点和终点,0,1,2,3标记,分别代表上右下左边界上的起点或终端
//    int tmp_dir[4][2] = {{0,rand()%n},{rand()%n,n-1},{n-1,rand()%n},{rand()%n,0}};
    int **tmp_dir = new int*[4];
    for(int i =0; i<4; ++i)
    {
        if(i==0)
        {
            tmp_dir[i] = new int[2];
            tmp_dir[i][0] = 0;
            tmp_dir[i][1] = rand()%n;
        }
        else if(i==1)
        {
            tmp_dir[i] = new int[2];
            tmp_dir[i][0] = rand()%n;
            tmp_dir[i][1] = n-1;
        }
        else if(i==2)
        {
            tmp_dir[i] = new int[2];
            tmp_dir[i][0] = n-1;
//            do
//            {
            tmp_dir[i][1] = (tmp_dir[i-2][1]+n/2)%n;
//            }
//            while(abs(tmp_dir[i][1]-tmp_dir[i-2][1])>n)||(n<=4)||(n%2==0)||(n>=110))        //输入非整型错误
    {
        cout<<" * 输入迷宫维数应当为整型奇数(4<维度大小<110):";
        cin.clear(); // 清除输入流错误标记
        cin.sync();// 取走刚才输入流中的字符
    }
    do
    {
        complexMaze_Maker();     //生成复杂迷宫
    }
    while(mazeTransformer()==ERROR);

//    testInt();
    /**
    迷宫展示
    **/
    char choice;
    while(1)
    {
        system("cls");      //清屏
        cout<<"\n\t\t\t\t电脑复杂迷宫展示界面\n"<>choice;
        while(choice!='1'&&choice!='2'&&choice!='3')        //判断选项是否为规定值
        {
            if(choice>'9'||choice<'0')      //输入类型错误提示
            {
                cout<<" * 警告!输入类型错误!!"<>choice;
            }
            else        //输入值错误提示
            {
                cout<<" * 请输入正确选项并按回车:";
                cin>>choice;
            }
        }
        switch(choice)      //进入电脑生成迷宫模块的功能函数
        {
        case '1':
            solveMaze_Maker(mazeStart[0],mazeStart[1]); //回溯生成通路
            cout<<"迷宫的通路:"<>choice;
        while(choice!='1'&&choice!='2'&&choice!='3')        //判断选项是否为规定值
        {
            if(choice>'9'||choice<'0')      //输入类型错误提示
            {
                cout<<" * 警告!输入类型错误!!"<>choice;
            }
            else        //输入值错误提示
            {
                cout<<" * 请输入正确选项并按回车:";
                cin>>choice;
            }
        }
        switch(choice)      //进入电脑生成迷宫模块的功能函数
        {
        case '1':
            resetMazeData();    //释放指针,重置程序
            return OK;
        case '2':
            resetMazeData();    //释放指针,重置程序
            return RESTART;
        case '3':
            resetMazeData();    //释放指针
            exit(0);
        }
    }
    return OK;
}
/************************************************************************\
函数名称:
    computerMenu
功能描述:
    电脑自动生成迷宫的菜单栏目界面展示
函数参数:
    无
返回值:
    执行成功:OK
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
\************************************************************************/
Status computerMenu()        //电脑生成迷宫模块子菜单界面函数
{
    char choice;
    while(1)
    {
        system("cls");      //清屏
        cout<<"\n\t\t\t\t电脑生成迷宫菜单\n"<>choice;
        while(choice!='1'&&choice!='2'&&choice!='3'&&choice!='4')        //判断选项是否为规定值
        {
            if(choice>'9'||choice<'0')      //输入类型错误提示
            {
                cout<<" * 警告!输入类型错误!!"<>choice;
            }
            else        //输入值错误提示
            {
                cout<<" * 请输入正确选项并按回车:";
                cin>>choice;
            }
        }
        int SimComMaze_Maker_Status;  //返回简单生成迷宫的返回值
        int CompComMaze_Maker_Status;   //返回复杂生成迷宫的返回值
        switch(choice)      //进入电脑生成迷宫模块的功能函数
        {
        case '1':
            SimComMaze_Maker_Status = simpleMaze_Menu();      //电脑简单迷宫展示界面
            if(SimComMaze_Maker_Status==RESTART)  //返回主菜单,重置程序
            {
                return OK;
            }
            else if(SimComMaze_Maker_Status==OK)  //再来一个迷宫
            {
                break;
            }
            break;
        case '2':
            CompComMaze_Maker_Status = complexMaze_Menu();      //电脑复杂迷宫展示界面
            if(CompComMaze_Maker_Status==RESTART)  //返回主菜单,重置程序
            {
                return OK;
            }
            else if(CompComMaze_Maker_Status==OK)  //再来一个迷宫
            {
                break;
            }
            break;
        case '3':
            return OK;
        case '4':
            exit(0);
        }
    }
}
/************************************************************************\
函数名称:
    artiMaze_Maker
功能描述:
    人工迷宫输入器和生成器
函数参数:
    无
返回值:
    执行成功:OK
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
    Yaung于2023年4月2日修改本模块,email: [email protected]
    修改原因:增加对于输入迷宫形式错误的处理
\************************************************************************/
Status artiMaze_Maker()       //人工迷宫输入器和生成器
{
    resetMazeData();
    /**
    接受用户输入的信息并且等待转化为迷宫
    **/
    initMazeStro();    //初始化生成迷宫储存空间
    /*
    例子:
    5
    1 1 1 1 1
    2 0 0 0 1
    1 1 0 1 1
    1 1 0 0 3
    1 1 1 1 1
    6
    1 1 1 1 1 1
    2 0 0 0 1 1
    1 1 0 0 0 1
    1 1 1 1 0 1
    1 1 0 0 0 1
    1 1 3 1 a 1
    */
    cout<<" * 输入迷宫每行的符号:"<>tmp))        //输入非整型错误
            {
                cout<<" * 警告,输入类型错误!:"<4):";
    while(!(cin>>n)||(n<=4))        //输入非整型错误
    {
        cout<<" * 输入迷宫维数应当为大于4的整型数字:";
        cin.clear(); // 清除输入流错误标记
        cin.sync();// 取走刚才输入流中的字符
    }
    int status_input;
    int status_artiMaze;
    do
    {
        status_input = artiMaze_Maker();   //输入迷宫
        if(status_input == ERROR)
        {
            continue;
        }

        status_artiMaze = mazeTransformer();
        if(status_artiMaze==ERROR)
        {
            cout<<"输入迷宫格式有误!请重新输入!"<>choice;
        while(choice!='1'&&choice!='2'&&choice!='3')        //判断选项是否为规定值
        {
            if(choice>'9'||choice<'0')      //输入类型错误提示
            {
                cout<<" * 警告!输入类型错误!!"<>choice;
            }
            else        //输入值错误提示
            {
                cout<<" * 请输入正确选项并按回车:";
                cin>>choice;
            }
        }
        int status_solveMaze;   //迷宫的求解状态
        switch(choice)      //进入电脑生成迷宫模块的功能函数
        {
        case '1':
            status_solveMaze = solveMaze_Maker(mazeStart[0],mazeStart[1]);
            if(status_solveMaze==OK) //回溯生成通路
            {
                mazeDisplay();  //展示迷宫通路
            }
            else if(status_solveMaze==DENY)
            {
                cout<>choice;
        while(choice!='1'&&choice!='2'&&choice!='3')        //判断选项是否为规定值
        {
            if(choice>'9'||choice<'0')      //输入类型错误提示
            {
                cout<<" * 警告!输入类型错误!!"<>choice;
            }
            else        //输入值错误提示
            {
                cout<<" * 请输入正确选项并按回车:";
                cin>>choice;
            }
        }
        switch(choice)      //进入电脑生成迷宫模块的功能函数
        {
        case '1':
            resetMazeData();    //释放指针,重置程序
            return OK;
        case '2':
            resetMazeData();    //释放指针,重置程序
            return RESTART;
        case '3':
            resetMazeData();    //释放指针
            exit(0);
        }
    }
}
/************************************************************************\
函数名称:
    artificicalMenu
功能描述:
    人工生成迷宫的菜单栏目界面展示与输入格式说明
函数参数:
    无
返回值:
    执行成功:OK
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
\************************************************************************/
Status artificicalMenu()       //人工生成迷宫菜单函数
{
    {
        char choice;

        while(1)
        {
            system("cls");      //清屏
            cout<<"\n\t\t\t\t人工生成迷宫菜单\n"<>choice;
            while(choice!='1'&&choice!='2'&&choice!='3')        //判断选项是否为规定值
            {
                if(choice>'9'||choice<'0')      //输入类型错误提示
                {
                    cout<<" * 警告!输入类型错误!!"<>choice;
                }
                else        //输入值错误提示
                {
                    cout<<" * 请输入正确选项并按回车:";
                    cin>>choice;
                }
            }
            int artiMaze_Maker_Status;
            switch(choice)      //进入电脑生成迷宫模块的功能函数
            {
            case '1'://自己输入迷宫生成器
                artiMaze_Maker_Status = artiMaze_Menu();
                if(artiMaze_Maker_Status==RESTART)  //返回主菜单,重置程序
                {
                    return OK;
                }
                else if(artiMaze_Maker_Status==OK)  //再来一个迷宫
                {
                    break;
                }
            case '2':
                return OK;
            case '3':
                exit(0);
            }
        }
    }
}
/************************************************************************\
函数名称:
    main
功能描述:
    迷宫问题系统主菜单界面展示
函数参数:
    无
返回值:
    正常:0
模块  :
    Yaung于2023年3月3日创建本模块,email: [email protected]
    Yaung于2023年5月4日修改本模块,email: [email protected]
        修改原因:增加对于编码的设置使得cmd也能正常运行
\************************************************************************/
int main()//#腾讯会议:529-875-839
{
    //主函数
//    system("chcp 65001");   //编码设置
    srand((unsigned)time(NULL));    //设置运行程序时间为随机种子
    char choice;         //选项值
    do
    {
        system("cls");      //清屏
        cout<<"\n\t\t\t\t迷宫问题主菜单\n"<>choice;
        while(choice!='1'&&choice!='2'&&choice!='3')        //判断选项是否为规定值
        {
            if(choice>'9'||choice<'0')      //输入类型错误提示
            {
                cout<<" * 警告!输入类型错误!!"<>choice;
            }
            else        //输入值错误提示
            {
                cout<<" * 请输入正确选项并按回车:";
                cin>>choice;
            }
        }
        switch(choice)
        {
        case '1':
            computerMenu();     //电脑生成迷宫菜单
            break;
        case '2':
            artificicalMenu();      //自己生成迷宫菜单
            break;
        }
    }
    while(choice!='3');
    return 0;
}

----以上为个人思考与见解,有误请指点,有想法也可联系交流!

               ~~~~~~~~~~~~~~               谢谢观看,点赞是对作者的最大支持!!

你可能感兴趣的:(程序设计,软件工程,设计模式,c++,c语言,动态规划,算法,数据结构)