迷宫求解-数据结构-课业设计

合集下载

数据结构课程设计迷宫问题求解

数据结构课程设计迷宫问题求解

数据结构课程设计迷宫问题求解正文:一、引言在数据结构课程设计中,迷宫问题求解是一个经典且常见的问题。

迷宫问题求解是指通过编程实现在迷宫中找到一条从起点到终点的路径。

本文将详细介绍如何用数据结构来解决迷宫问题。

二、问题分析1.迷宫定义:迷宫是由多个格子组成的矩形区域,其中包括起点和终点。

迷宫中的格子可以是墙壁(无法通过)或者通道(可以通过)。

2.求解目标:在给定的迷宫中,找到从起点到终点的一条路径。

3.输入:迷宫的大小、起点坐标、终点坐标以及墙壁的位置。

4.输出:从起点到终点的路径,或者提示无解。

三、算法设计1.基础概念a) 迷宫的表示:可以使用二维数组来表示迷宫,数组的元素可以是墙壁、通道或者路径上的点。

b) 坐标系统:可以使用(x, y)来表示迷宫中各个点的坐标。

c) 方向定义:可以用上、下、左、右等四个方向来表示移动的方向。

2.深度优先搜索算法(DFS)a) 算法思想:从起点开始,沿着一个方向一直走到无法继续为止,然后回退到上一个点,再选择其他方向继续探索。

b) 算法步骤:i) 标记当前点为已访问。

ii) 判断当前点是否为终点,如果是则返回路径;否则继续。

iii) 遍历四个方向:1.如果该方向的下一个点是通道且未访问,则继续向该方向前进。

2.如果该方向的下一个点是墙壁或已访问,则尝试下一个方向。

iv) 如果四个方向都无法前进,则回退到上一个点,继续向其他方向探索。

3.广度优先搜索算法(BFS)a) 算法思想:从起点开始,逐层向外探索,直到找到终点或者所有点都被访问。

b) 算法步骤:i) 标记起点为已访问,加入队列。

ii) 循环以下步骤直到队列为空:1.取出队首元素。

2.判断当前点是否为终点,如果是则返回路径;否则继续。

3.遍历四个方向:a.如果该方向的下一个点是通道且未访问,则标记为已访问,加入队列。

iii) 如果队列为空仍未找到终点,则提示无解。

四、算法实现1.选择合适的编程语言和开发环境。

迷宫求解数据结构课程设计

迷宫求解数据结构课程设计

迷宫求解数据结构课程设计《数据结构》课程设计题目:迷宫求解班级:学号:作者姓名:指导教师:2012年12月11日目录1.需求分析 (1)2.概要设计 (1)2.1.数据结构 (1)2.1.1.逻辑结构 (1)2.1.1.存储结构 (2)2.2.基本操作 (3)2.2.1.迷宫中栈的基本操作 (3)2.2.2.迷宫的抽象数据类型 (3)2.2.3.本程序包含三个模块 (4)3.详细设计 (5)4.调试与分析 (9)5.用户手册 (9)6. 测试结果 (10)7. 附录 (12)8. 参考文献 (12)9、心得体会 (12)10、小组成员工作分配 (13)一、需求分析(1)以二维数组maze[n+2][m+2]表示迷宫,其中:maze[0][j]和maze[n+1][j](0<=j<=m+1)及maze[i][0]和maze[i][m+1](0<=i<=j+1)为添加的一圈障碍。

数组中以元素值为0的表示通路,1表示障碍,限定迷宫的大小,m,n<=0。

(2)迷宫的入口位置和出口位置可由用户自行设定。

(3)如设定的迷宫处在通路,则值输出迷宫中的通路,即0,现实的0连起来就是一个迷宫通路路径;如设定的迷宫中不出在通路,则输出“该迷宫找不到通路!”;(4)测试样例:输入迷宫的长宽为5和6,输入迷宫为:1 0 0 1 1 10 0 1 1 1 11 0 0 0 1 10 1 0 1 1 11 1 0 0 0 0当入口位置为(1,2),出口位置为(5,6)时,则输出数据为:# # # # # # # ## 0 ## 0 ## 0 0 ## 0 ## 0 0 0 0 ## # # # # # # #(5)程序执行的命令为:1、输入迷宫的尺寸;2、创建迷宫;3、输入迷宫的的出入口位置;4、求解迷宫;输出迷宫的解。

二、概要设计2.1.数据结构2.1.1、逻辑结构1)栈的定义:限定仅在表尾进行插入或删除操作的线性表;2)操作特性:后进先出;3) ADT定义:ADT Stack{数据对象:D={ai|ai∈CharSet,i=1,2,…,n,n>=0}数据关系:R1={<ai-1,ai>|ai-1,ai∈D,i=2,…,n}基本操作:InitStack(&S)操作结果:构造一个空栈S。

数据结构迷宫求解课程设计报告

数据结构迷宫求解课程设计报告

课程设计报告课程名称:数据结构报告题目:迷宫求解学生姓名:XX所在学院:信息科学与工程专业班级:软件工程学生学号:XXXXXXXXXXX指导教师:XXX课程设计任务书摘要本程序主若是求迷宫中从人口到出口的所有途径是一个经典的程序设计问题。

运算机解迷宫时,通经常使用的是“穷举求解”的方式,即从入口动身,顺某一方向向前探讨,假设能走通,那么继续往前走;不然沿原路返回,换一个方向在继续探讨,直至所有可能的通路都探讨完为止。

当前位置“可通”,那么纳入“当前途径”,并继续朝“下一名置”探讨,即切换为“下一名置”为“当前位置”,如此重复直至抵达出口;假设当前位置“不可通”,那么应顺着“来的方向”退回到“前一通道块”,假设该通道块的周围4个方块均“不可通”那么应从当前途径删除该通道块。

所谓“下一名置”指的是“当前位置”周围4个方向(东、南、西、北)上相邻的方块。

以栈S来记录“当前途径”,那么栈顶中寄存的是“当前途径上最后一个通道块”。

因此即为“当前途径入栈”;“从当前途径上删除前一通道块”为“出栈”。

在那个进程中能够输出迷宫所走通的途径,在这次课程设计中迷宫是由数组预先概念好的,不能由用户概念生成,能够加入随机函数,自动生成二维数组,还能够用户自己输入迷宫。

关键词:栈;存储结构;数组目录目录 (3)一、课题分析 (1)二、需求分析 (1)1. 主模块功能描述 (1)2. 子程序模块设计 (1)三、设计方案 (1)1.类设计 (1)int point_x; 序模块设计 (2)设计方案与实施与整体设计思想 (2)3. 主模块 (3)子模块 (4)主菜单 (5)功能A模块 (5)功能B模块 (5)功能C模块 (6)四、详细设计 (7)1.用结构体构建栈 (7)SElemType *top; 建类 (7)int point_x; 数组构建一个迷宫 (7)case 1: 戏移动操纵指令 (8)}戏移动操纵指令四个方向的实现 (8)else if ((map[point_x - 1][point_y] == 0 && direction != 5) || map[point_x - 1][point_y] == 3) 戏移动进程中假设是未找到通路,需要返回的指令 (9)}戏的主循环 (9)}戏的开始于终止 (10)if (map[point_x][point_y] == 3) 宫地图的查看与通关后的途径查询 (10)五、设计总结 (11)结论与心得 (11)五、参考文献 (12)一、课题分析(1)该题目为迷宫求解。

数据结构课程设计-迷宫问题

数据结构课程设计-迷宫问题

数据结构课程设计-迷宫问题正文:一、引言本文档旨在设计一个解决迷宫问题的数据结构课程项目。

迷宫问题是一个典型的寻路问题,要求从起点出发,在迷宫中找到一条路径到达终点。

迷宫由多个房间组成,这些房间之间通过门相连。

二、问题描述迷宫问题包含以下要素:1.迷宫的拓扑结构:迷宫由多个房间和门组成,每个房间有四面墙壁,每面墙壁可能有门或者是封闭的。

迷宫的起点和终点是预先确定的。

2.寻路算法:设计一个算法,在迷宫中找到一条从起点到终点的路径。

路径的选择标准可以是最短路径、最快路径或者其他约束条件。

3.可视化展示:实现一个可视化界面,在迷宫中展示起点、终点、路径,用于直观地演示解决方案。

三、设计思路1.数据结构设计:选择合适的数据结构来表示迷宫和路径,例如使用二维数组或者图来表示迷宫的拓扑结构,使用栈或队列来辅助寻路算法的实现。

2.寻路算法设计:可以使用深度优先搜索、广度优先搜索、Dijkstra算法、A算法等经典算法来实现寻路功能。

根据实际需求选择最合适的算法。

3.可视化展示设计:使用图形界面库(如Tkinter、Qt等)创建迷宫展示窗口,并实时更新迷宫的状态、路径的变化。

可以通过颜色、动画等方式增加交互性。

四、实现步骤1.创建迷宫:根据预设的迷宫大小,使用数据结构来创建对应的迷宫数据。

2.设定起点和终点:在迷宫中选择起点和终点的位置,将其标记出来。

3.寻路算法实现:根据选择的寻路算法,在迷宫中找到一条路径。

4.可视化展示:使用图形界面库创建窗口,并将迷宫、起点、终点、路径等信息展示出来。

5.更新迷宫状态:根据算法实现的过程,实时更新迷宫中的状态,并将路径显示在迷宫上。

附件:1.代码实现:包含迷宫创建、寻路算法实现和可视化展示的源代码文件。

2.演示视频:展示项目实际运行效果的视频文件。

法律名词及注释:1.数据结构:指在计算机科学中定义和组织数据的方式和方式的基础设施。

2.寻路算法:用于解决寻找路径的问题的算法。

迷宫游戏数据结构课程设计

迷宫游戏数据结构课程设计

迷宫游戏数据结构课程设计
1、简介
本文档旨在设计一个迷宫游戏的数据结构课程项目,通过使用合适的数据结构和算法,实现一个能够自动和解决迷宫的程序。

本项目将使用C++语言来实现。

2、功能需求
本项目的主要功能如下:
- 自动一个迷宫地图
- 实现玩家在迷宫地图中的移动
- 实现迷宫的解决算法
3、技术方案
本项目将采用以下技术方案来实现功能:
3.1 迷宫算法
为了一个随机的迷宫地图,我们将采用深度优先搜索(DFS)算法或者随机Prim算法来迷宫。

这些算法可以保证的迷宫是连通的且没有死胡同。

3.2 玩家移动
玩家将使用键盘输入来控制移动,通过获取键盘输入来实现玩
家在迷宫中的移动。

游戏将使用图形界面来呈现迷宫和玩家的位置。

3.3 迷宫解决算法
迷宫解决算法将使用广度优先搜索(BFS)算法或者深度优先搜
索(DFS)算法来搜索迷宫的路径。

该算法将从起点出发,逐步搜索
迷宫的每个可达点,直到找到终点或者遍历完整个迷宫。

4、开发计划
本项目的开发计划如下:
1、确定项目需求和技术方案 - 2天
2、实现迷宫算法 - 3天
3、实现玩家移动功能 - 2天
4、实现迷宫解决算法 - 3天
5、创建图形界面 - 2天
6、进行测试和调试 - 3天
7、完善文档和准备演示 - 2天
5、附件
本文档没有附件。

6、法律名词及注释
本文档没有涉及任何法律名词及注释。

数据结构课程设计 迷宫求解

数据结构课程设计  迷宫求解

考虑使用一个二维数组表示迷宫.所有的通路用0表示,墙用1表示,出口用9表示,入口用6表示,已经过点用3表示.输出走出迷宫的过程.从这个问题的求解过程中可以简单总结出两个算法,一是探路过程,二是输出路线.1.探路过程探路过程算法可归纳为:[1]从入口位置开始,检查东西南北四个方向上的通路,如果发现出口则成功退出,否则将所有通路坐标压入栈;[2]从栈中取出一个坐标,将其标记为当前位置(标记数字3),再次判断通路情况;[3]如此进行,直到发现出口则成功退出,若栈空而且未发现出口,则失败退出.这里使用到的回溯过程可描述为: 每到达一点时,会将所有可能的通路坐标(标记数字0的节点)压入栈.所以当到达一点,而不存在可能的通路时,自然没有相应的坐标压入栈,而此时便从栈中取出上一个点所压入的可能的一个通路坐标,并继续作通路判断,这便是一个回溯的过程.2.输出某一较短路线将所有在探路过程中经过的点(标记数字3的节点)按实际探路路线存入队列,对头为入口,队尾为出口.这些点可能存在绕路的情况,所以可用下面的算法输出某一较短路线.[1]将队尾(出口)节点设置为当前判断节点;[2]从当前判断节点(x,y)的前驱节点开始,向前遍历队列,如果发现相邻节点(其坐标可以为(x+1,y),(x-1,y),(x,y+1),(x,y-1)之一),则删除该相临节点至当前判断节点的前驱节点之间的所有节点;[3]将该相临节点设置为当前判断节点,继续判断相临节点;[4]当当前判断节点为对头节点时退出.该算法所得到的路线不一定是最短路线,想得到最短路线,可考虑使用树结构将所有由出口至入口的路线保留为一子树,树高最短的子树即为最短路线.但此算法可保证所得路线不会存在绕路情况.3.表示节点坐标的类public class MazeCell {private int x, y;//表示x轴y轴坐标public MazeCell() {}public MazeCell(int i, int j) {x = i;y = j;}public boolean equals(Object o) {if (!(o instanceof MazeCell))return false;MazeCell cell = (MazeCell) o;return cell.x == x && cell.y == y;}public String toString() {return x + "," + y;}public int getX() {return x;}public void setX(int x) {this.x = x;}public int getY() {return y;}public void setY(int y) {this.y = y;}}4.所使用的栈数据结构import java.util.LinkedList;public class Stack<T> {private LinkedList<T> storage = new LinkedList<T>(); /** 入栈*/public void push(T v) {storage.addFirst(v);}/** 出栈,但不删除*/public T peek() {return storage.getFirst();}/** 出栈*/public T pop() {return storage.removeFirst();}/** 栈是否为空*/public boolean empty() {return storage.isEmpty();}/** 打印栈元素*/public String toString() {return storage.toString();}}5.求解迷宫问题import java.io.BufferedReader;import java.io.IOException;import java.io.InputStreamReader;import java.io.PrintStream;import java.util.Iterator;import java.util.LinkedList;import java.util.List;import java.util.ListIterator;public class Maze {private int rows = 0, cols = 0;// 迷宫的行数与列数private char[][] store, path;// 迷宫矩阵private MazeCell currentCell, exitCell = new MazeCell(),entryCell = new MazeCell();// 当前节点,出口节点,入口节点private static final char EXIT = '9', ENTRY = '6', VISITED = '3';// 出口标记,入口标记,已经过节点标记private static final char PASS = '0', W ALL = '1';// 通路标记,墙标记private Stack<MazeCell> mazeStack = new Stack<MazeCell>();// 探路过程所使用栈private List<MazeCell> currentList = new LinkedList<MazeCell>();// 路经的路线队列public Maze() {// 构造迷宫int row = 0, col = 0;Stack<String> mazeRows = new Stack<String>();InputStreamReader isr = new InputStreamReader(System.in);BufferedReader buffer = new BufferedReader(isr);System.out.println("Enter a rectangular maze using the following" + " characters: \n6-entry\n9-exit\n1-wall\n0-passage\n"+ "Enter one line at a time; end with Ctrl-d;");try {String str = buffer.readLine();while (str != null) {row++;cols = str.length();str = "1" + str + "1";mazeRows.push(str);if (str.indexOf(EXIT) != -1) {exitCell.setX(row);exitCell.setY(str.indexOf(EXIT));}if (str.indexOf(ENTRY) != -1) {entryCell.setX(row);entryCell.setY(str.indexOf(ENTRY));}str = buffer.readLine();}} catch (IOException e) {e.printStackTrace();}rows = row;store = new char[rows + 2][];store[0] = new char[cols + 2];for (; !mazeRows.empty(); row--)store[row] = (mazeRows.pop()).toCharArray();store[rows + 1] = new char[cols + 2];for (col = 0; col <= cols + 1; col++) {store[0][col] = WALL;store[rows + 1][col] = WALL;}path = new char[rows + 2][];copyArray(store, path);}/** 二维数组复制*/private void copyArray(char[][] src, char[][] tar) {for (int i = 0; i < src.length; i++) {tar[i] = new char[cols + 2];for (int j = 0; j < src[i].length; j++)tar[i][j] = src[i][j];}}/** 二维数组输出*/private void display(PrintStream out, char[][] carray) {for (int row = 0; row <= rows + 1; row++)out.println(carray[row]);out.println();}/** 将未访问并可通路的节点压入栈*/private void pushUnvisited(int row, int col) {if (store[row][col] == PASS || store[row][col] == EXIT) mazeStack.push(new MazeCell(row, col));}/** 探路过程*/public void exitMaze(PrintStream out) {currentCell = entryCell;currentList.add(currentCell);out.println();while (!currentCell.equals(exitCell)) {int row = currentCell.getX();int col = currentCell.getY();display(System.out, store);if (!currentCell.equals(entryCell))store[row][col] = VISITED;pushUnvisited(row - 1, col);pushUnvisited(row + 1, col);pushUnvisited(row, col - 1);pushUnvisited(row, col + 1);if (mazeStack.empty()) {display(out, store);out.println("Failure");return;} else {currentCell = mazeStack.pop();currentList.add(currentCell);}}display(out, store);out.println("Success");}/** 得到某一输出路线*/private void getPath() {if (currentList.size() <= 0)return;MazeCell cell = currentList.get(currentList.size() - 1);while (cell != currentList.get(0)) {List<MazeCell> subList = currentList.subList(0, currentList.indexOf(cell));ListIterator<MazeCell> itr = subList.listIterator();while (itr.hasNext()) {MazeCell target = itr.next();if (adjoin(cell, target)) {removeElements(currentList.indexOf(target) + 1, currentList .indexOf(cell));cell = target;break;}}}}/** 删除队列中由from至to的连续元素*/private void removeElements(int from, int to) {int turn = to - from;while (turn > 0) {currentList.remove(from);turn--;}}/** 判断两个节点是否相邻*/private boolean adjoin(MazeCell current, MazeCell target) {if ((current.getX() == target.getX() + 1 || current.getX() == target.getX() - 1)&& (current.getY() == target.getY()))return true;if ((current.getY() == target.getY() + 1 || current.getY() == target.getY() - 1)&& (current.getX() == target.getX()))return true;return false;}/** 输出路线*/public void printPath(PrintStream out) {getPath();out.println("Path:");if (currentList.size() >= 2) {currentList.remove(currentList.size() - 1);currentList.remove(0);}Iterator<MazeCell> itr = currentList.iterator();while (itr.hasNext()) {MazeCell cell = itr.next();path[cell.getX()][cell.getY()] = VISITED;}display(System.out, path);}public static void main(String[] args) {Maze maze = new Maze();maze.exitMaze(System.out);maze.printPath(System.out);}}6.结果输出Enter a rectangular maze using the following characters: 6-entry9-exit1-wall0-passageEnter one line at a time; end with Ctrl-d;90000110110000000600//构造的迷宫如下111111119000011110111100000110060011111111//开始探路11111111900001111011110000011006001 11111111111111 1900001 1110111 1000001 1006301 11111111111111 1900001 1110111 1000001 1006331 1111111 1111111 1900001 1110111 1000031 1006331 11111111111111 1900001 1110111 1000331 1006331 1111111 1111111 1900001 1110111 1003331 1006331 11111111111111 1900001 1110111 1033331 1006331 1111111111111119000011110111133333110063311111111111111119000011110111133333113063311111111111111119000011110111133333113363311111111//下一步为回溯过程111111119000011110111133333113363311111111111111119000011113111133333113363311111111111111119030011113111133333113363311111111111111119033011113111133333113363311111111111111119033311113111133333113363311111111//下一步为回溯过程111111119333311113111133333113363311111111SuccessPath:111111119330011113111100300110060011111111。

数据结构课程设计_迷宫求解

数据结构课程设计_迷宫求解

迷宫求解一.问题描述对迷宫问题的求解过程实际就是从入口开始,一步一步地走到出口的过程。

基本要求:输入一个任意大小的迷宫数据,用递归和非递归两种方法求出一条走出迷宫的路径,并将路径输出。

二.设计思路在本程序中用两种方法求解迷宫问题-非递归算法和递归算法。

对于非递归算法采用回溯的思想,即从入口出发,按某一方向向前探索,若能走通,并且未走过,则说明某处可以到达,即能到达新点,否则试探下一方向;若所有的方向均没有通路,或无路可走又返回到入口点。

在求解过程中,为了保证在到达某一点后不能向前继续行走(无路)时,能正确返回前一点以便继续从下一个方向向前试探,则需要用一个栈保存所能到达的没一点的下标与该点前进的方向,然后通过对各个点的进出栈操作来求得迷宫通路。

对于递归算法,在当前位置按照一定的策略寻找下个位置,在下个位置又按照相同的策略寻找下下个位置…;直到当前位置就是出口点,每一步的走法都是这样的。

随着一步一步的移动,求解的规模不断减小;如果起始位置是出口,说明路径找到,算法结束,如果起始位置的四个方向都走不通,说明迷宫没有路径,算法也结束。

另外,为了保证迷宫的每个点都有四个方向可以试探,简化求解过程,将迷宫四周的值全部设为1,因此将m行n列的迷宫扩建为m+2行,n+2列,同时用数组来保存迷宫阵列。

三.数据结构设计在迷宫阵列中每个点都有四个方向可以试探,假设当前点的坐标(x,y),与其相邻的四个点的坐标都可根据该点的相邻方位而得到,为了简化问题,方便求出新点的坐标,将从正东开始沿顺时针进行的这四个方向的坐标增量放在一个结构数组move[4]中,每个元素有两个域组成,其中x为横坐标增量,y为纵坐标增量,定义如下:typedef struct{int x,y;}item;为到达了某点而无路可走时需返回前一点,再从前一点开始向下一个方向继续试探。

因此,还要将从前一点到本点的方向压入栈中。

栈中的元素由行、列、方向组成,定义如下:typedef struct{int x,y,d;}DataType;由于在非递归算法求解迷宫的过程中用到栈,所以需定义栈的类型,本程序中用的是顺序栈,类型定义如下;typedef struct{DataType data[MAXSIZE];int top;}SeqStack, *PSeqStack;四.功能函数设计(1)函数PSeqStack Init_SeqStack()此函数实现对栈的初始化工作。

迷宫求解课程设计报告

迷宫求解课程设计报告

迷宫求解课程设计报告一、课程目标知识目标:1. 让学生掌握迷宫问题的基础知识,理解迷宫的构成元素及求解方法。

2. 培养学生运用数据结构表示迷宫,了解并运用深度优先搜索、广度优先搜索等算法解决迷宫问题。

技能目标:1. 培养学生运用计算机编程语言实现迷宫求解算法,提高编程能力。

2. 培养学生通过分析迷宫问题,设计合理的解决方案,并运用算法进行求解。

情感态度价值观目标:1. 培养学生对计算机科学产生兴趣,增强学习积极性。

2. 培养学生面对问题勇于挑战、积极思考的良好品质。

3. 培养学生团队合作意识,学会在团队中分工合作,共同解决问题。

课程性质分析:本课程为计算机科学相关课程,以迷宫问题为载体,教授数据结构、算法等知识。

课程注重理论与实践相结合,强调学生的动手实践能力。

学生特点分析:本课程面向的学生为初中年级学生,他们具备一定的计算机操作基础,对新鲜事物充满好奇,但可能对复杂算法的理解和运用存在一定难度。

教学要求:1. 教师应注重理论与实践相结合,通过实例讲解,使学生更容易理解和掌握知识。

2. 教学过程中,注重启发式教学,引导学生主动思考,培养学生的创新意识。

3. 针对不同学生的特点,因材施教,使学生在掌握基本知识的基础上,提高自身能力。

二、教学内容根据课程目标,教学内容分为以下三个部分:1. 迷宫基础知识- 迷宫的构成元素与类型- 迷宫问题的数学模型2. 迷宫求解算法- 数据结构:图、队列、栈- 深度优先搜索算法- 广度优先搜索算法- 最短路径算法:Dijkstra算法、A*算法3. 编程实践- 编程语言:Python、C++等- 迷宫求解算法的实现- 迷宫求解算法的优化教学大纲安排如下:第一周:- 迷宫基础知识学习- 数据结构图、队列、栈的介绍第二周:- 深度优先搜索算法与广度优先搜索算法讲解- 课堂练习:运用算法解决迷宫问题第三周:- 最短路径算法Dijkstra算法、A*算法讲解- 编程实践:实现迷宫求解算法第四周:- 编程实践:优化迷宫求解算法- 学生作品展示与评价教材章节关联:本教学内容与教材中“图与搜索算法”章节相关,涉及到的知识点包括图的基本概念、搜索算法及其应用。

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

设计题目:迷宫问题求解问题描述迷宫问题是取自心理学的一个古典实验。

在该实验中,把一只老鼠从一个无顶大盒子的门放入,在盒子中设置了许多墙,对行进方向形成了多处阻挡。

盒子仅有一个出口,在出口处放置一块奶酪,吸引老鼠在迷宫中寻找道路以到达出口。

对同一只老鼠重复进行上述实验,一直到老鼠从入口走到出口,而不走错一步。

老鼠经过多次试验最终学会走通迷宫的路线。

设计一个计算机程序对任意设定的矩形迷宫如下图A所示,求出一条从入口到出口的通路,或得出没有通路的结论。

图A●任务:●功能要求:(1) 建立一个大小为m×n的任意迷宫(迷宫数据可由用户输入或由程序自动生成),并在屏幕上显示出来;(2)找出一条通路的二元组(i,j)数据序列,(i,j)表示通路上某一点的坐标。

(3)用一种标志(如数字8)在迷宫中标出该条通路;(4)在屏幕上输出迷宫和通路;(5)上述功能可用菜单选择。

需求分析:1.迷宫的建立:迷宫中存在通路和障碍,为了方便迷宫的创建,可用0表示通路,用1表示障碍,这样迷宫就可以用0、1矩阵来描述,2.迷宫的存储:迷宫是一个矩形区域,可以使用二维数组表示迷宫,这样迷宫的每一个位置都可以用其行列号来唯一指定,但是二维数组不能动态定义其大小,我们可以考虑先定义一个较大的二维数组maze[M+2][N+2],然后用它的前m行n列来存放元素,即可得到一个m×n的二维数组,这样(0,0)表示迷宫入口位置,(m-1,n-1)表示迷宫出口位置。

注:其中M,N分别表示迷宫最大行、列数,本程序M、N的缺省值为39、39,当然,用户也可根据需要,调整其大小。

3.迷宫路径的搜索:首先从迷宫的入口开始,如果该位置就是迷宫出口,则已经找到了一条路径,搜索工作结束。

否则搜索其上、下、左、右位置是否是障碍,若不是障碍,就移动到该位置,然后再从该位置开始搜索通往出口的路径;若是障碍就选择另一个相邻的位置,并从它开始搜索路径。

为防止搜索重复出现,则将已搜索过的位置标记为2,同时保留搜索痕迹,在考虑进入下一个位置搜索之前,将当前位置保存在一个队列中,如果所有相邻的非障碍位置均被搜索过,且未找到通往出口的路径,则表明不存在从入口到出口的路径。

这实现的是广度优先遍历的算法,如果找到路径,则为最短路径。

以矩阵0 0 1 0 1 为例,来示范一下1 0 0 1 01 0 0 0 10 0 1 0 0首先,将位置(0,0)(序号0)放入队列中,其前节点为空,从它开始搜索,其标记变为2,由于其只有一个非障碍位置,所以接下来移动到(0,1)(序号1),其前节点序号为0,标记变为2,然后从(0,1)移动到(1,1)(序号2),放入队列中,其前节点序号为1,(1,1)存在(1,2)(序号3)、(2,1)(序号4)两个可移动位置,其前节点序号均为2.对于每一个非障碍位置,它的相邻非障碍节点均入队列,且它们的前节点序号均为该位置的序号,所以如果存在路径,则从出口处节点的位置,逆序就可以找到其从出口到入口的通路。

如下表所示:0 1 2 3 4 5 6 7 8 9 10由此可以看出,得到最短路径:(3,4)(3,3)(2,3)(2,2)(1,2)(1,1)(0,1)(0,0)搜索算法流程图如下所示:测试数据:0表示可以行走的区域,1表示不可行走的区域。

概要设计1.①构建一个二维数组maze[M+2][N+2]用于存储迷宫矩阵②自动或手动生成迷宫,即为二维数组maze[M+2][N+2]赋值③构建一个队列用于存储迷宫路径④建立迷宫节点struct point,用于存储迷宫中每个节点的访问情况⑤实现搜索算法⑥屏幕上显示操作菜单2.本程序包含10个函数:(1)主函数main()(2)手动生成迷宫函数shoudong_maze()(3)自动生成迷宫函数zidong_maze()(4)将迷宫打印成图形print_maze()(5)打印迷宫路径(若存在路径) result_maze()(6)入队enqueue()(7)出队dequeue()(8)判断队列是否为空is_empty()(9)访问节点visit()(10)搜索迷宫路径mgpath()实现概要设计中定义的所有数据类型及操作的伪代码算法1.节点类型和指针类型迷宫矩阵类型:int maze[M+2][N+2];为方便操作使其为全局变量迷宫中节点类型及队列类型:struct point{int row,col,predecessor} que[512]2.迷宫的操作(1)手动生成迷宫void shoudong_maze(int m,int n){定义i,j为循环变量for(i<=m)for(j<=n)输入maze[i][j]的值}(2)自动生成迷宫void zidong_maze(int m,int n){定义i,j为循环变量for(i<=m)for(j<=n)maze[i][j]=rand()%2 //由于rand()产生的随机数是从0到RAND_MAX,RAND_MAX是定义在stdlib.h中的,其值至少为32767),要产生从X到Y的数,只需要这样写:k=rand()%(Y-X+1)+X;}(3)打印迷宫图形void print_maze(int m,int n){用i,j循环变量,将maze[i][j]输出□、■}(4)打印迷宫路径void result_maze(int m,int n){用i,j循环变量,将maze[i][j]输出□、■、☆}(5)搜索迷宫路径①迷宫中队列入队操作void enqueue(struct point p){将p放入队尾,tail++}②迷宫中队列出队操作struct point dequeue(struct point p){head++,返回que[head-1]}③判断队列是否为空int is_empty(){返回head==tail的值,当队列为空时,返回0}④访问迷宫矩阵中节点void visit(int row,int col,int maze[41][41]){建立新的队列节点visit_point,将其值分别赋为row,col,head-1,maze[row][col]=2,表示该节点以被访问过;调用enqueue(visit_point),将该节点入队}⑤路径求解void mgpath(int maze[41][41],int m,int n){先定义入口节点为struct point p={0,0,-1},从maze[0][0]开始访问。

如果入口处即为障碍,则此迷宫无解,返回0 ,程序结束。

否则访问入口节点,将入口节点标记为访问过maze[p.row][p.col]=2,调用函数enqueue(p)将该节点入队。

判断队列是否为空,当队列不为空时,则运行以下操作:{ 调用dequeue()函数,将队头元素返回给p,如果p.row==m-1且p.col==n-1,即到达出口节点,即找到了路径,结束如果p.col+1<n且maze[p.row][p.col+1]==0,说明未到迷宫右边界,且其右方有通路,则visit(p.row,p.col+1,maze),将右边节点入队标记已访问如果p.row+1<m且maze[p.row+1][p.col]==0,说明未到迷宫下边界,且其下方有通路,则visit(p.row+1,p.col,maze),将下方节点入队标记已访问如果p.col-1>0且maze[p.row][p.col-1]==0,说明未到迷宫左边界,且其左方有通路,则visit(p.row,p.col-1,maze),将左方节点入队标记已访问如果p.row-1>0且maze[p.row-1][p.col]==0,说明未到迷宫上边界,且其上方有通路,则visit(p.row,p.col+1,maze),将上方节点入队标记已访问}访问到出口(找到路径)即p.row==m-1且p.col==n-1,则逆序将路径标记为3即maze[p.row][p.col]==3;while(p.predecessor!=-1){p=queue[p.predecessor]; maze[p.row][p.col]==3;}最后将路径图形打印出来。

3.菜单选择while(cycle!=(-1))☆手动生成迷宫请按:1☆自动生成迷宫请按:2☆退出请按:3scanf("%d",&i);switch(i){case 1:请输入行列数(如果超出预设范围则提示重新输入)shoudong_maze(m,n);print_maze(m,n);mgpath(maze,m,n);if(X!=0) result_maze(m,n);case 2:请输入行列数(如果超出预设范围则提示重新输入)zidong_maze(m,n);print_maze(m,n);mgpath(maze,m,n);if(X!=0) result_maze(m,n);case 3:cycle=(-1);break;}详细设计:#include"stdlib.h"#include"stdio.h"#define N 39#define M 39int X;int maze[N+2][M+2];struct point{int row,col,predecessor;}queue[512];int head=0,tail=0;void shoudong_maze(int m,int n){int i,j;printf("\n\n");printf("请按行输入迷宫,0表示通路,1表示障碍:\n\n");for(i=0;i<m;i++)for(j=0;j<n;j++)scanf("%d",&maze[i][j]);}void zidong_maze(int m,int n){int i,j;printf("\n迷宫生成中……\n\n");system("pause");for(i=0;i<m;i++)for(j=0;j<n;j++)maze[i][j]=rand()%2;//由于rand()产生的随机数是从0到RAND_MAX//RAND_MAX是定义在stdlib.h中的,其值至少为32767)//要产生从X到Y的数,只需要这样写:k=rand()%(Y-X+1)+X;}void print_maze(int m,int n){int i,j;printf("\n迷宫生成结果如下:\n\n");printf("迷宫入口\n");printf("↓");for(i=0;i<m;i++){printf("\n");for(j=0;j<n;j++){if(maze[i][j]==0) printf("□");if(maze[i][j]==1) printf("■");}}printf("→迷宫出口\n");}void result_maze(int m,int n){int i,j;printf("迷宫通路(用☆表示)如下所示:\n\t");for(i=0;i<m;i++){printf("\n");for(j=0;j<n;j++){if(maze[i][j]==0||maze[i][j]==2) printf("□");if(maze[i][j]==1) printf("■");if(maze[i][j]==3) printf("☆");}}}void enqueue(struct point p){queue[tail]=p;tail++;}struct point dequeue(){head++;return queue[head-1];}int is_empty(){return head==tail;}void visit(int row,int col,int maze[41][41]){ struct point visit_point={row,col,head-1};maze[row][col]=2;enqueue(visit_point);}int mgpath(int maze[41][41],int m,int n){X=1;struct point p={0,0,-1};if(maze[p.row][p.col]==1){printf("\n===============================================\n");printf("此迷宫无解\n\n");X=0;return 0;}maze[p.row][p.col]=2;enqueue(p);while(!is_empty()){p=dequeue();if((p.row==m-1)&&(p.col==n-1)) break;if((p.col+1<n)&&(maze[p.row][p.col+1]==0)) visit(p.row,p.col+1,maze);if((p.row+1<m)&&(maze[p.row+1][p.col]==0)) visit(p.row+1,p.col,maze);if((p.col-1>=0)&&(maze[p.row][p.col-1]==0)) visit(p.row,p.col-1,maze);if((p.row-1>=0)&&(maze[p.row-1][p.col]==0)) visit(p.row-1,p.col,maze);}if(p.row==m-1&&p.col==n-1){printf("\n=================================================== ===============\n");printf("迷宫路径为:\n");printf("(%d,%d)\n",p.row,p.col);maze[p.row][p.col]=3;while(p.predecessor!=-1){p=queue[p.predecessor];printf("(%d,%d)\n",p.row,p.col);maze[p.row][p.col]=3;}}else{printf("\n======================================================= ======\n");printf("此迷宫无解!\n\n");X=0;}return 0;}void main(){int i,m,n,cycle=0;while(cycle!=(-1)){printf("********************************************************************************\n");printf(" 欢迎进入迷宫求解系统\n");printf(" 设计者:苏浩然\n"); printf("********************************************************************************\n"); printf(" ☆手动生成迷宫请按:1\n");printf(" ☆自动生成迷宫请按:2\n");printf(" ☆退出请按:3\n\n");printf("********************************************************************************\n"); printf("\n");printf("请选择你的操作:\n");scanf("%d",&i);switch(i){case 1:printf("\n请输入行数:");scanf("%d",&m);printf("\n");printf("请输入列数:");scanf("%d",&n);while((m<=0||m>39)||(n<=0||n>39)){printf("\n抱歉,你输入的行列数超出预设范围(0-39,0-39),请重新输入:\n\n");printf("请输入行数:");scanf("%d",&m);printf("\n");printf("请输入列数:");scanf("%d",&n);}shoudong_maze(m,n);print_maze(m,n);mgpath(maze,m,n);if(X!=0) result_maze(m,n);printf("\n\nPress Enter Contiue!\n");getchar();while(getchar()!='\n');break;case 2:printf("\n请输入行数:");scanf("%d",&m);printf("\n");printf("请输入列数:");scanf("%d",&n);while((m<=0||m>39)||(n<=0||n>39)){printf("\n抱歉,你输入的行列数超出预设范围(0-39,0-39),请重新输入:\n\n");printf("请输入行数:");scanf("%d",&m);printf("\n");printf("请输入列数:");scanf("%d",&n);}zidong_maze(m,n);print_maze(m,n);mgpath(maze,m,n);if(X!=0) result_maze(m,n);printf("\n\nPress Enter Contiue!\n");getchar();while(getchar()!='\n');break;case 3:cycle=(-1);break;default:printf("\n");printf("你的输入有误!\n");printf("\nPress Enter Contiue!\n");getchar();while(getchar()!='\n');break;}}}●测试结果:●用户手册:。

相关文档
最新文档