【数据结构】迷宫问题报告+源码C/C++

之前帮别人写的一个报告,是关于栈的迷宫问题。内容不多,代码在最后。分享给大家,喜欢可以点赞+关注。原创无偿分享,勿商用。

迷宫求解

设计目的

仅认识到栈是一种特殊的线性表是远远不够的,本次实习的目的在于使学生深入了解栈的特征,以便在实际问题背景下灵活运用它,同时还将巩固这种数据结构的构造方法。

问题描述

迷宫问题是取自心理学的一个古典实验。在该实验中,把一只老鼠从一个无顶大盒子的门放入,在盒子中设置了许多墙,对行进方向形成了多处阻挡。盒子仅有一个出口,在出口处放置一块奶酪,吸引老鼠在迷宫中寻找道路以到达出口。对同一只老鼠重复进行上述实验,一直到老鼠从入口走到出口,而不走错一步。老鼠经过多次试验最终学会走通迷宫的路线。设计一个计算机程序对任意设定的矩形迷宫如下图A所示,求出一条从入口到出口的通路,或得出没有通路的结论。

图A

 函数说明

  1. 对栈的处理

InitStack(&S)

操作结果:构造一个空栈 S。

StackEmpty(S)

操作结果:若 S 为空栈,则返回 TRUE,否则返回 FALSE。

Push(&S, e)初始条件:栈 S 已存在。

操作结果:在栈 S 的栈顶插入新的栈顶元素 e。

Pop(&S,&e)

操作结果:删除 S 的栈顶元素,并以 e 返回其值。

  1. 对迷宫的处理

InitMaze(&maze,a,row,col)

操作结果:对迷宫进行初始化。构成迷宫的字符型数组,以空白字符表示通路,以字符 ‘#’表示障碍,并在迷宫四周加上一圈障碍。

PrintMaze(&maze)

操作结果:以字符形式输出迷宫。

MazePath(&maze,start,end)

操作结果:若迷宫 M 中存在一条通路,则按以下规定改变迷宫 M 的状态:以字符’

-’表示路径上的位置,字符‘X’表示“死胡同”,否则迷宫的状态不变。

程序流程图

  

重要算法说明

从入口位置开始,将其入栈。

判断当前位置是否可达到,可达到则入栈。

对东西南北四个方向进行遍历,判断能否通过,可通过则更改当前位置。

若当前位置为死路,则出栈并做出‘X’标识。

若当前位置为出口,则存下路径长度退出循环。

使用说明

在该界面,用户有4个可选选项。

选择创建迷宫选项后,需给出迷宫的行列,后用0,1分别表示通路和障碍。

选择查看迷宫选项后,可以查看当前的字符迷宫。

选择寻找通路选项后,可以查看当前迷宫的通路。

调试报告

  1. 用户手动输入的数组,后自动转化为字符迷宫。

2.查询当前的字符迷宫

3.求解路径后输出迷宫的路径

心得体会

本次实验采用的栈的思想对迷宫问题进行了解决。采用了菜单的方式提供4个功能,分别为创建迷宫、查看迷宫、寻找通路以及退出。对于寻找通路功能来说若可以达到终点,则此算法会按东南西北的顺序寻找到终点的通路,无法保证该通路为最短路,但一定可以走到终点。

本次实验中的另一个的核心就是如何建立存储迷宫的数据结构,即如何用链栈来存储迷宫路径中的各位置以及下一步将要走的方向。并且对试探无路可走后回退的位置要做特殊标识,另外还要在迷宫中标识走过的位置,避免在有路可走时还走回头路出现死循环。算法PrintMaze 和 MazePath 的空间复杂度均为 O(m*n),故本次实验的空间复杂度也为 O(m*n)

附录:

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

typedef int Status;

#define FALSE 0

#define TRUE 1

#define RANGE 20 //RANGE 为实际分配的空间行列数

int N,M,maze[50][50];

typedef struct { //表达迷宫元素位置信息的坐标

int r,c;

} PosType;

typedef struct { //m,n 为待处理的迷宫的行列数,RANGE 为实际分配的空间行列数

int m,n;

char arr[RANGE][RANGE];

} MazeType;

typedef int directiveType;//东南西北方向用 1,2,3,4 整数对应

typedef struct { //路径拟用栈来存储,此处定义栈元素中数据域的信息

int step;//存储到达该点时经历的步数

PosType seat;//该点位置

directiveType op;//从该点位置走向下一位置的方向

} ElemType;

typedef struct NodeType { //路径拟用链栈来存储

ElemType data;

struct NodeType *next;

} NodeType,*LinkType;

typedef struct { //对链栈的头指针和元素个数进行封装

LinkType top;

int size;

} Stack;

//以上是对存储结构逐层进行定义

void InitStack(Stack &S) { //构建空链栈,不带头结点

S.top=NULL;

S.size=0;

}

Status MakeNode(LinkType &p,ElemType e) { //创建一个新结点,以便插入,本函数可作为链式存储创建结点的通用函数,

p=(NodeType *)malloc(sizeof(NodeType));

if(!p)

return FALSE;

p->data=e;

p->next=NULL;

return TRUE;

}

Status Push(Stack &S,ElemType e) { //入栈操作,栈头(链表头)进行插入

LinkType p;

if(MakeNode(p,e)) {

p->next=S.top;

S.top=p;

S.size++;

return TRUE;

}

return FALSE;

}

Status StackEmpty(Stack S) { //判断是否为空栈,这里是通过 top 指针为 NULL 来判断的,也可以通过 size 是否为 0 来判断

if(S.top==NULL)

return TRUE;

return FALSE;

}

Status Pop(Stack &S,ElemType &e) { //出栈操作,删除表头元素

LinkType p;

if(StackEmpty(S))

return FALSE;

else {

p=S.top;

S.top=S.top->next;

e=p->data;

S.size--;

free(p);

return TRUE;

}

}

Status pass(MazeType maze,PosType now_pos) { //判断迷宫 Maze 中,当前位置 now_pos 是否是一个可达位置

if(maze.arr[now_pos.r][now_pos.c]==' ')

return TRUE;

return FALSE;

}

Status Same(PosType now_pos,PosType end) { //判断当前位置 now_pos 是否已达出口

if(now_pos.r==end.r && now_pos.c==end.c)

return TRUE;

return FALSE;

}

void FootPrint(MazeType &maze,PosType now_pos) {

//在迷宫中标识走过的位置,避免在有路可走时还走回头路出现死循环

maze.arr[now_pos.r][now_pos.c]='-';

}

PosType NextPos(PosType now_pos,int op) {

//通过 op 的值,确定下一步的位置,下一步位置实际是当前位置的四个邻居中的一个

switch(op) {

case 1:

now_pos.c++; //向东走

break;

case 2:

now_pos.r++; //向南走

break;

case 3:

now_pos.c--; //向西走

break;

case 4:

now_pos.r--; //向北走

break;

}

return now_pos;

}

void MarkPrint(MazeType &maze,PosType p) {

//对试探无路可走后回退的位置做特殊标识

maze.arr[p.r][p.c]='X';

}

void PrintMaze(MazeType maze) {

//对迷宫输出,实际是对一个二维数组的输出

int i,j;

printf("\n");

for(i=0; i<maze.m; i++) {

printf("\t");

for(j=0; j<maze.n; j++) {

printf("%c ",maze.arr[i][j]);

}

printf("\n");

}

printf("\n");

}

void InitMaze(MazeType &maze,int a[][50],int row,int col) {

//根据二维数组来初始化迷宫,这个二维数组可以设计为由用户从键盘输入,

//控制每行长度的实际就是定义列的数值,所以要明确参数N

int i,j;

maze.m=row;

maze.n=col;

for(i=0; i<row; i++)

for(j=0; j<col; j++) {

if(a[i][j]==0)

maze.arr[i][j]=' ';

else

maze.arr[i][j]='#';

}

}

int MazePath(MazeType &maze,PosType start,PosType end) { //返回值为路径的长度,返回 0 表示无通路

Stack s;

int now_step=1; //统计路径长度

int found=0;

ElemType e; //以栈元素的形式暂存当前位置的相关信息,以便入栈构成路径

PosType now_pos=start; //设当前位置为开始位置

InitStack(s);

do { //栈不空且未到出口则继续循环

if(pass(maze,now_pos)) { //如果 now_pos 位置可达则先入栈

FootPrint(maze,now_pos); //如果可通则标记为 -,后边如果发现是死胡同,则会重新标记为 X

e.step=now_step;

e.seat=now_pos;

e.op=1;

Push(s,e);

if(Same(now_pos,end)) {

found=s.size;

} else {

now_pos=NextPos(now_pos,1); //到新位置时默认先向东走

now_step++;

}

} else if(!StackEmpty(s)) {

Pop(s,e); //如果 now_pos 位置不可达,且栈不空,则把刚入栈的元素弹出做相关判断

while((e.op==4) && !StackEmpty(s)) {

MarkPrint(maze,e.seat); //标识此路不通

Pop(s,e); //回到上一个位置

now_step--; //不减一的话可以用于统计走过的总步数

}

if(e.op<4) { //如果还有方向未走过,则沿新的方向继续试探

e.op++;

Push(s,e); //默认新方向的下一位置可达,将当前位置入栈

now_pos=NextPos(e.seat,e.op); //通过当前位置,以及去下一位置的方向得出新的位置,再循环查看新位置是否可达

}

}

} while(!StackEmpty(s) && !found);

return found;

}

void Print(int maze[][50]) {

int i,j;

printf("表示迷宫的数组\n");

for(i=0; i<M; i++) {

printf("\t");

for(j=0; j<N; j++) {

printf("%d ",maze[i][j]);

}

printf("\n");

}

printf("\n");

}

void showMenu() {

system("cls");

putchar('\n');

printf("          ");

printf("********************************************************************");

putchar('\n');

putchar('\n');

printf("                    ");

printf("1--创建一个新的迷宫\n");

putchar('\n');

printf("                    ");

printf("2--查看当前迷宫\n");

putchar('\n');

printf("                    ");

printf("3--寻找当前迷宫的通路\n");

putchar('\n');

printf("                    ");

printf("4--退出\n");

putchar('\n');

printf("          ");

printf("********************************************************************\n");

}

int main() {

int step=0;

int f=0;

MazeType L;

while(1) {

int ch;

PosType start,end;

showMenu();

printf("请选择功能:");

scanf("%d",&ch);

switch(ch) {

case 1:

printf("请输入迷宫矩阵的行与列!\n");

scanf("%d %d",&M,&N); //M为迷宫的行 N为迷宫的列

printf("请使用0,1分别代表通道与障碍输入%d行%d列的迷宫矩阵!(输入时使用空格分隔)\n",M,N);

int i,j;

int op;

scanf("%d",&op);

if(op==1) {

int i,j;

for(i=0; i<M; i++)//随机生成0,1迷宫矩阵

for(j=0; j<N; j++) {

if(rand()%10000>=6666) maze[i][j]=1;

else maze[i][j]=0;

}

f=1;

InitMaze(L,maze,M,N);//调用 InitMaze 函数将二维数组初始化为迷宫

printf("迷宫创建完成!");

system("pause");

break;

case 2:

if(!f) {

printf("当前不存在迷宫,请先创建一个新的迷宫!");

system("pause");

break;

} else {

printf("由数组转化出的迷宫");

PrintMaze(L);//输出已转化的迷宫

system("pause");

break;

}

case 3:

if(!f) {

printf("当前不存在迷宫,请先创建一个新的迷宫!");

system("pause");

break;

} else {

PrintMaze(L);

MazeType L_tp;

InitMaze(L_tp,maze,M,N);//调用 InitMaze 函数将二维数组初始化为迷宫

printf("请输入迷宫的入口坐标!\n");

scanf("%d %d",&start.r,&start.c); //设置迷宫的入口

start.r--;start.c--;

printf("请输入迷宫的出口坐标!\n");

scanf("%d %d",&end.r,&end.c); //设置迷宫的出口

end.r--;end.c--;

if(L_tp.arr[start.r][start.c]=='#'||L_tp.arr[end.r][end.c]=='#'){

printf("输入异常!起点或终点为障碍......");

system("pause");

break;

}

if(step==MazePath(L_tp,start,end))

printf("迷宫的路径,其中S为起点,F为终点,路径用-表示,路径长度为:%d",step);

else

printf("此迷宫没有通路!");

L_tp.arr[start.r][start.c]='S';

L_tp.arr[end.r][end.c]='F';

PrintMaze(L_tp);

system("pause");

break;

}

case 4:

return 0;

}

}

}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/233514.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Linux基础】3. 文件基本属性

文章目录 【 1. 文件的属主和属组 】【 2. 显示文件的类型、权限 】2.1 文件类型2.2 文件权限 【 3. 更改文件属性 】3.1 chgrp 更改文件属组3.2 chown 更改文件所有者3.3 更改文件权限3.3.1 数字法更改文件权限3.3.2 符号法更改文件权限 【 1. 文件的属主和属组 】 Linux 系统…

Vue的脚手架

脚手架配置 脚手架文档&#xff1a;Vue CLI npm config set registry https://registry.npm.taobao.org vue.config.js配置选项&#xff1a; 配置参考 | Vue CLI ref选项 ref和id类似&#xff0c;给标签打标识。 document.getElementById(btn); this.$ref.btn; 父子组…

Kafka 分级存储在腾讯云的实践与演进

导语 腾讯云消息队列 Kafka 内核负责人鲁仕林为大家带来了《Kafka 分级存储在腾讯云的实践与演进》的精彩分享&#xff0c;从 Kafka 架构遇到的问题与挑战、Kafka 弹性架构方案类比、Kafka 分级存储架构及原理以及腾讯云的落地与实践四个方面详细分享了 Kafka 分级存储在腾讯云…

基于JAVA的海南旅游景点推荐系统 开源项目

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 用户端2.2 管理员端 三、系统展示四、核心代码4.1 随机景点推荐4.2 景点评价4.3 协同推荐算法4.4 网站登录4.5 查询景点美食 五、免责说明 一、摘要 1.1 项目介绍 基于VueSpringBootMySQL的海南旅游推荐系统&#xff…

STM32 RTC总结

RTC入侵检测Tamper RTC Tamper功能就是&#xff0c;MCU在Tamper管脚检测到一个指定边缘信号&#xff08;可配置&#xff09;时&#xff0c;就主动清除所有备份寄存器数据的功能。如果需要&#xff0c;可以使能Tamper中断&#xff0c;在每次检测到Tamper信号后执行指定代码。 在…

用23种设计模式打造一个cocos creator的游戏框架----(二十一)组合模式

1、模式标准 模式名称&#xff1a;组合模式 模式分类&#xff1a;结构型 模式意图&#xff1a;将对象组合成树型结构以表示“部分-整体”的层次结构。Composite 使得用户对单个对象和组合对象的使用具有一致性。 结构图&#xff1a; 适用于&#xff1a; 1、想表示对象的部分…

算法通关村第十关—快速排序(青铜)

快速排序 快排的基本过程 快速排序是将分治法运用到排序问题的典型例子  快速排序基本思想是&#xff1a;通过一个标记pivot元素将n个元素的序列划分为左右两个子序列left和right,.其中left中的元素都比pivot小&#xff0c;right的都比pivot的大&#xff0c;然后再次对Ieft和r…

关于折线回归

一、说明 今天的帖子主要是关于使用折线回归找到最佳值。即将某条曲线分解成包络线段&#xff0c;然后用分段回归方式优化。但它也涉及使用 SAS 和 R 的剂量反应研究和样条曲线。这不是第一篇关于这些主题的文章&#xff0c;但我确实想在其中添加折线。只是因为它还在使用。 二…

使用docker-compose搭建docker私服与配置WebUI

简介 本文介绍了使用docker compose 搭建 docker私服 环境 Docker version 24.0.6, build ed223bc Docker Compose version v2.21.0 正文 一、创建registry文件夹 我的路径是/usr/loca/docker/registry 二、创建并编写docker-compose.yml version: "3.9" services…

ios备忘录怎么导入华为 方法介绍

作为一个常常需要在不同设备间切换的人&#xff0c;我深知备忘录的重要性。那些突如其来的灵感、重要的会议提醒、甚至是生活中的琐碎小事&#xff0c;我们都习惯性地记录在备忘录里。但当我决定从iPhone转向华为时&#xff0c;一个问题困扰了我&#xff1a;如何将那些珍贵的备…

智能优化算法应用:基于学校优化算法3D无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于学校优化算法3D无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于学校优化算法3D无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.学校优化算法4.实验参数设定5.算法结果6.…

Axure中继器完成表格的增删改查的自定义元件(三列表格与十列表格)

目录 一、中继器 1.1 定义 1.2 特点 1.3 适用场景 二、三列表格增删改查 2.1 实现思路 2.2 效果演示 三、十列表格增删改查 3.1 实现思路 3.2 效果演示 一、中继器 1.1 定义 在Axure中&#xff0c;"中继器"通常指的是界面设计中的一个元素&#xff0c;用…

IntelliJ IDE 插件开发 | (三)消息通知与事件监听

系列文章 IntelliJ IDE 插件开发 |&#xff08;一&#xff09;快速入门IntelliJ IDE 插件开发 |&#xff08;二&#xff09;UI 界面与数据持久化IntelliJ IDE 插件开发 |&#xff08;三&#xff09;消息通知与事件监听 前言 在前两篇文章中讲解了关于插件开发的基础知识&…

【linux】解决matplotlib中文显示乱码

一、Debian操作系统版本号 $ cat /etc/debian_version 12.4 二、问题 matplotlib中文显示乱码。 三、解决方法&#xff1a; plt.rcParams["font.sans-serif"] "Noto Sans CJK JP" 四、解决过程 1.问GPT 回答了一些相关问题&#xff0c;但是好像用…

Python tkinter 初探Toplevel控件搭建父子窗口

目录 Toplevel控件搭建父子窗口 最简明的父子窗口框架 改进一&#xff1a;屏蔽和开放按钮 改进二&#xff1a;子窗口始终在主窗口之上 改进三&#xff1a;增加子窗口的关闭协议 改进四&#xff1a;使子窗口长获焦点 总结 Toplevel控件搭建父子窗口 最近&#xff0c;用P…

k8s集群1.23.0版本部署说明

1.部署 k8s1.23.0版本与1.26.0版本的部署基本差不多&#xff0c;只不过k8s 1.23版本不需要部署cri-docker&#xff0c;所以只需要在1.26.0版本部署的基础上不要cri-docker的部署即可 参考&#xff1a;kubeadm部署k8s 1.26.0版本高可用集群_kubeadm 高可用集群-CSDN博客 搭建…

行为型设计模式(一)模版方法模式 迭代器模式

模板方法模式 Template 1、什么是模版方法模式 模版方法模式定义了一个算法的骨架&#xff0c;它将其中一些步骤的实现推迟到子类里面&#xff0c;使得子类可以在不改变算法结构的情况下重新定义算法中的某些步骤。 2、为什么使用模版方法模式 封装不变部分&#xff1a;模版…

搭建知识付费平台?明理信息科技为你提供全程解决方案

明理信息科技saas知识付费平台 在当今数字化时代&#xff0c;知识付费已经成为一种趋势&#xff0c;越来越多的人愿意为有价值的知识付费。然而&#xff0c;公共知识付费平台虽然内容丰富&#xff0c;但难以满足个人或企业个性化的需求和品牌打造。同时&#xff0c;开发和维护…

【容器Docker】Docker学习笔记

1、什么是Docker&#xff1a; Docker 将程序和程序运行所依赖的所有环境都打包到镜像里。“build once, run anywhere”Docker 是容器的一种实现。 Windows 下如何安装Docker: 官方安装教程&#xff1a;Install Docker Desktop on Windows | Docker Docs有两种安装套装&…

使用openMVS库,在VS2022中启用c++17标准编译仍然报错

使用openMVS库&#xff0c;在VS2022中启用c17标准编译仍然报错 现象 项目中引用了某些开源库&#xff08;例如openmvs2.1.0&#xff09;&#xff0c;编译时要求启用编译器对c17的支持。 没问题&#xff01;大家都知道在下图所示的位置调整C语言标准&#xff1a; 但是&#…