C语言数据结构实现:迷宫问题的通用解法
yuyutoo 2024-10-23 16:43 19 浏览 0 评论
问题:
以一个m*n的长方阵表示迷宫,0和1分别表示迷宫中的通路和障碍。设计一个程序,对任意设定的迷宫,求出一条从入口到出口的通路,或得出没有通路的结论。
基本要求
输入的形式和范围:
非递归:行列为整型,坐标为整型
递归:迷宫以整型二维数组形式输入
输出的形式:非递归输出三元组通路和方阵通路;
递归以方阵输出迷宫和所有通路;
1、非递归算法,求一条通路输出三元组形式如:(1,1,1),(1,2,2),(2,2,2),(3,2,3),(3,1,2),…和方阵通路;
2、递归算法,求得迷宫中所有可能的通路,以方阵形式输出迷宫及其通路。
递归解法:
#include <stdio.h>
#include <malloc.h>
#define M 6
#define N 6
#define END N-2
int flag=0;
typedef struct
{
int x,y,d;
}position;
/*创建迷宫*/
void creat_maze(int a[][M])
{
int i,j;
for(i=0;i<=N-1;i++)
for(j=0;j<=M-1;j++)
scanf("%1d",&a[i][j]);
}
position nextq(int maze[][M],position q)
{
if(q.d==0)
q.y++;
else if(q.d==1)
q.x++;
else if(q.d==2)
q.y--;
else if(q.d==3)
q.x--;
return q;
}
void initialmat(int route[][M]) //初始化输出路径图
{
int i,j;
for(i=0;i<N;i++)
for(j=0;j<M;j++)
route[i][j]=1;
}
void print_maze(int mat[][M]) //输出迷宫
{
int i,j;
for(i=0;i<N;i++)
{
for(j=0;j<M;j++)
printf("%d ",mat[i][j]);
printf("\n");
}
}
void MazePath(int maze[][M],int route[][M],position ps)
{
position q;
route[ps.x][ps.y]=0;
maze[ps.x][ps.y]=-1;
q=nextq(maze,ps);
q.d=0;
if(q.x==END&&q.y==END)
{
flag++;
printf("\n第%d条路:\n",flag);
route[END][END]=0;
print_maze(route);
route[END][END]=1;
}
else if(maze[q.x][q.y]==0)
MazePath(maze,route,q);
if(ps.d<=3)
{
ps.d++;
MazePath(maze,route,ps);
}
route[ps.x][ps.y]=1;
maze[ps.x][ps.y]=0;
}
void main()
{
int maze[N][M],route[N][M];
position ps;
ps.x=ps.y=1;
ps.d=0;
printf("输入一个迷宫(%d*%d):\n",N,M);
creat_maze(maze);
printf("\n输出该迷宫:\n");
print_maze(maze);
printf("输出从(1,1)到(%d,%d)的所有通路:\n",N-2,M-2);
initialmat(route);
MazePath(maze,route,ps);
}
非递归解法:
#include<stdio.h>
#include<stdlib.h>
#define Elemtype int
#define MAXSIZE 50
typedef struct
{
int x,y;
}mark; //起点、终点坐标
typedef struct
{
Elemtype x,y; //迷宫数组坐标(x,y)
int d; //下一步的方向
}TriMatrix;
typedef struct LStackNode
{
TriMatrix elem;
struct LStackNode *next;
}LStack,*PLStack;
//栈的基本操作
PLStack InitStack(PLStack S)
{
S=NULL;
return S;
}
int StackEmpty(PLStack S)
{
if(S==NULL) return 1;
else return 0;
}
PLStack Push(PLStack S,TriMatrix e) //入栈
{
PLStack P;
P=(PLStack)malloc(sizeof(LStack));
P->elem=e;
P->next=S;
S=P;
return S;
}
PLStack Delete(PLStack S) //删除栈头
{
PLStack P;
P=S;
if(!StackEmpty(S))
{
S=S->next;
free(P);
return S;
}
}
TriMatrix Pop(PLStack S,TriMatrix e) //出栈
{
if(!StackEmpty(S))
{
e=S->elem;
return e;
}
}
void MazePath(mark start,mark end,
Elemtype maze[MAXSIZE][MAXSIZE],
int m,int n,int diradd[4][2]) //寻找路径
{
int i,j,d;
int a,b;
TriMatrix elem,e;
PLStack S1,S2;
S1=InitStack(S1);
S2=InitStack(S2);
maze[start.x][start.y]=2; //修改入口坐标,做标记
elem.x=start.x;
elem.y=start.y;
elem.d=0; //开始为0
S1=Push(S1,elem);
while(!StackEmpty(S1)) //栈不空,有路可走
{
elem=Pop(S1,elem);
S1=Delete(S1);
i=elem.x;
j=elem.y;
d=elem.d+1;
while(d<=4)
{
a=i+diradd[d-1][0];
b=j+diradd[d-1][1];
if(a==end.x&&b==end.y&&maze[a][b]==0) //出口
{
elem.x=i;
elem.y=j;
elem.d=d;
S1=Push(S1,elem);
elem.x=a;
elem.y=b;
elem.d=0; //方向输出为0,判断是否为出口
S1=Push(S1,elem);
printf("\n1=东 2=南 3=西 4=北 0为走出迷宫\n\n路径为:(行坐标,列坐标,方向)\n");
while(S1) //逆置序列 并输出路径
{
e=Pop(S1,e);
S1=Delete(S1);
S2=Push(S2,e);
}
while(S2)
{
e=Pop(S2,e);
S2=Delete(S2);
printf("-->(%d,%d,%d)",e.x,e.y,e.d);maze[e.x][e.y]=3;
}
printf("\n\n");
for(i=0;i<=m+1;i++){
for(j=0;j<=n+1;j++)
if(maze[i][j]==3) printf("0 ");
else printf("1 ");
printf("\n");
}
return;
}//if
if(maze[a][b]==0)
{
maze[a][b]=2;
elem.x=i;
elem.y=j;
elem.d=d;
S1=Push(S1,elem);
i=a; j=b; d=0;
}
d++;
}
}
printf("没有出迷宫的路径\n");
}
//建立迷宫
void initmaze(int maze[MAXSIZE][MAXSIZE],int m,int n)
{
int i,j;
for(i=1;i<=m;i++)
for(j=1;j<=n;j++)
maze[i][j]=rand()%2;
for(i=0;i<=m+1;i++)
{
maze[i][0]=maze[i][n+1]=1;
}
for(j=0;j<=n+1;j++)
{
maze[0][j]=maze[m+1][j]=1;
}
printf("输出迷宫:\n");
for(i=0;i<=m+1;i++)
{
for(j=0;j<=n+1;j++)
printf("%d ",maze[i][j]);
printf("\n");
}
}
void main()
{
int maze[MAXSIZE][MAXSIZE];
mark start,end;
int m,n; //迷宫行列
printf("输入迷宫的行数m和列数n:\n");
scanf("%d%d",&m,&n);
int add[4][2]={{0,1},{1,0},{0,-1},{-1,0}};
initmaze(maze,m,n);
printf("输入入口坐标和出口坐标:\n");
scanf("%d%d%d%d",&start.x,&start.y,&end.x,&end.y);
MazePath(start,end,maze,m,n,add);
}
写在最后:对于准备学习C/C++编程的小伙伴,如果你想更好的提升你的编程核心能力(内功)不妨从现在开始!
编程学习书籍分享:
编程学习视频分享:
整理分享(多年学习的源码、项目实战视频、项目笔记,基础入门教程)
欢迎转行和学习编程的伙伴,利用更多的资料学习成长比自己琢磨更快哦!
对于C/C++感兴趣可以关注小编在后台私信我:【编程交流】一起来学习哦!可以领取一些C/C++的项目学习视频资料哦!已经设置好了关键词自动回复,自动领取就好了!
相关推荐
- 墨尔本一华裔男子与亚裔男子分别失踪数日 警方寻人
-
中新网5月15日电据澳洲新快网报道,据澳大利亚维州警察局网站消息,22岁的华裔男子邓跃(Yue‘Peter’Deng,音译)失踪已6天,维州警方于当地时间13日发布寻人通告,寻求公众协助寻找邓跃。华...
- 网络交友须谨慎!美国犹他州一男子因涉嫌杀害女网友被捕
-
伊森·洪克斯克(图源网络,侵删)据美国广播公司(ABC)25日报道,美国犹他州一名男子于24日因涉嫌谋杀被捕。警方表示,这名男子主动告知警局,称其杀害了一名在网络交友软件上认识的25岁女子。雷顿警...
- 一课译词:来龙去脉(来龙去脉 的意思解释)
-
Mountainranges[Photo/SIPA]“来龙去脉”,汉语成语,本指山脉的走势和去向,现比喻一件事的前因后果(causeandeffectofanevent),可以翻译为“i...
- 高考重要考点:range(range高考用法)
-
range可以用作动词,也可以用作名词,含义特别多,在阅读理解中出现的频率很高,还经常作为完形填空的选项,而且在作文中使用是非常好的高级词汇。...
- C++20 Ranges:现代范围操作(现代c++白皮书)
-
1.引言:C++20Ranges库简介C++20引入的Ranges库是C++标准库的重要更新,旨在提供更现代化、表达力更强的方式来处理数据序列(范围,range)。Ranges库基于...
- 学习VBA,报表做到飞 第二章 数组 2.4 Filter函数
-
第二章数组2.4Filter函数Filter函数功能与autofilter函数类似,它对一个一维数组进行筛选,返回一个从0开始的数组。...
- VBA学习笔记:数组:数组相关函数—Split,Join
-
Split拆分字符串函数,语法Split(expression,字符,Limit,compare),第1参数为必写,后面3个参数都是可选项。Expression为需要拆分的数据,“字符”就是以哪个字...
- VBA如何自定义序列,学会这些方法,让你工作更轻松
-
No.1在Excel中,自定义序列是一种快速填表机制,如何有效地利用这个方法,可以大大增加工作效率。通常在操作工作表的时候,可能会输入一些很有序的序列,如果一一录入就显得十分笨拙。Excel给出了一种...
- Excel VBA入门教程1.3 数组基础(vba数组详解)
-
1.3数组使用数组和对象时,也要声明,这里说下数组的声明:'确定范围的数组,可以存储b-a+1个数,a、b为整数Dim数组名称(aTob)As数据类型Dimarr...
- 远程网络调试工具百宝箱-MobaXterm
-
MobaXterm是一个功能强大的远程网络工具百宝箱,它将所有重要的远程网络工具(SSH、Telnet、X11、RDP、VNC、FTP、MOSH、Serial等)和Unix命令(bash、ls、cat...
- AREX:携程新一代自动化回归测试工具的设计与实现
-
一、背景随着携程机票BU业务规模的不断提高,业务系统日趋复杂,各种问题和挑战也随之而来。对于研发测试团队,面临着各种效能困境,包括业务复杂度高、数据构造工作量大、回归测试全量回归、沟通成本高、测试用例...
- Windows、Android、IOS、Web自动化工具选择策略
-
Windows平台中应用UI自动化测试解决方案AutoIT是开源工具,该工具识别windows的标准控件效果不错,但是当它遇到应用中非标准控件定义的UI元素时往往就无能为力了,这个时候选择silkte...
- python自动化工具:pywinauto(python快速上手 自动化)
-
简介Pywinauto是完全由Python构建的一个模块,可以用于自动化Windows上的GUI应用程序。同时,它支持鼠标、键盘操作,在元素控件树较复杂的界面,可以辅助我们完成自动化操作。我在...
- 时下最火的 Airtest 如何测试手机 APP?
-
引言Airtest是网易出品的一款基于图像识别的自动化测试工具,主要应用在手机APP和游戏的测试。一旦使用了这个工具进行APP的自动化,你就会发现自动化测试原来是如此简单!!连接手机要进行...
- 【推荐】7个最强Appium替代工具,移动App自动化测试必备!
-
在移动应用开发日益火爆的今天,自动化测试成为了确保应用质量和用户体验的关键环节。Appium作为一款广泛应用的移动应用自动化测试工具,为测试人员所熟知。然而,在不同的测试场景和需求下,还有许多其他优...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)