Java与算法之(5) - 老鼠走迷宫(深度优先算法)
小老鼠走进了格子迷宫,如何能绕过猫并以最短的路线吃到奶酪呢?
注意只能上下左右移动,不能斜着移动。
在解决迷宫问题上,深度优先算法的思路是沿着一条路一直走,遇到障碍或走出边界再返回尝试别的路径。
首先用一个二维数组来把迷宫“数字化”。
- int[][] maze = new int[5][4];
迷宫中每个格子的横纵坐标对应数组的一维和二维索引,例如最左上角的格子是maze[0][0],数组的值表示该格子是否可以通过,0表示可以通过,1表示该格子有猫。
初始化迷宫,标记猫的位置:
- this.maze[2][0] = 1;
- this.maze[1][2] = 1;
- this.maze[2][2] = 1;
- this.maze[3][2] = 1;
起点位置坐标是x=0,y=0,如果向右移动就是x=x+1,y=y,向下移动是x=x,y=y+1。我们预先规定每到一个格子都按照右、下、左、上的顺序尝试下一个格子是否能走,如果右边的格子没有猫且未出边界,就移动到下一个格子,继续按照右、下、左、上的顺序尝试;如果右边的格子不能走则尝试下面的格子。
下面这个二维数组用来遍历尝试四个方向的格子:
- int[][] next = new int[][] {
- {1, 0},
- {0, 1},
- {-1, 0},
- {0, -1}
- };
为了不走回头路,我们还需要另外一个二维数组标记哪些格子是已走过的,如果已走过则不能回头。
- int[][] mark = new int[5][4];
用一个栈记录路径
- LinkedList<Integer> map = new LinkedList<>();
走格子的思路是:
- for(遍历四个方向的格子) {
- if(格子超出边界 或 格子有猫 或 格子已经走过) {
- continue;
- } else {
- 移动到格子
- 记录当前格子已走过
- 记录当前路径
- for(以新格子为中心遍历四个方向的格子) {
- ......
- }
- }
- }
但是我们并不知道要走多少步才能到达目标,也就不知道循环要嵌套多少层,但是可以看出每次新的遍历循环开启后,执行的代码和上一层循环是一样的,所以这里用递归解决。来看完整的代码:
- import java.util.LinkedList;
- public class DfsRatMaze {
- int min = Integer.MAX_VALUE;
- int endX = 3; //目标点横坐标
- int endY = 3; //目标点纵坐标
- int width = 5; //迷宫宽度
- int height = 4; //迷宫高度
- int[][] maze = new int[5][4];
- int[][] mark = new int[5][4];
- LinkedList<Integer> map = new LinkedList<>();
- public void dfs(int startX, int startY, int step) {
- int[][] next = new int[][] { //按右->下->左->上的顺序尝试
- {1, 0},
- {0, 1},
- {-1, 0},
- {0, -1}
- };
- int nextX, nextY;
- int posible;
- if(startX == endX && startY == endY) {
- if(step < min)
- min = step;
- for(int i = map.size() - 1; i >= 0; i -= 2){
- nextX = map.get(i);
- nextY = map.get(i - 1);
- System.out.print("[" + nextX + "," + nextY + "]");
- if(i != 1)
- System.out.print("->");
- }
- System.out.println();
- return;
- }
- for(posible = 0; posible < next.length; posible++) { //按右->下->左->上的顺序尝试
- nextX = startX + next[posible][0];
- nextY = startY + next[posible][1];
- if(nextX < 0 || nextX >= width || nextY < 0 || nextY >= height) { //超出边界
- continue;
- }
- if(maze[nextX][nextY] == 0 && mark[nextX][nextY] == 0) { //非障碍且未标记走过
- map.push(nextX);
- map.push(nextY);
- mark[nextX][nextY] = 1;
- dfs(nextX, nextY, step + 1); //递归调用, 移动到下一格
- mark[nextX][nextY] = 0;
- map.pop();
- map.pop();
- }
- }
- }
- /*
- * 初始化迷宫
- */
- public void initMaze() {
- this.maze = new int[width][height];
- this.mark = new int[width][height];
- this.maze[2][0] = 1;
- this.maze[1][2] = 1;
- this.maze[2][2] = 1;
- this.maze[3][2] = 1;
- this.mark[0][0] = 1;
- //打印迷宫 _表示可通行 *表示障碍 !表示目标
- for(int y = 0; y < height; y++) {
- for(int x = 0; x < width; x++) {
- if(x == endX && y == endY) {
- System.out.print("! ");
- } else if(this.maze[x][y] == 1) {
- System.out.print("* ");
- } else {
- System.out.print("_ ");
- }
- }
- System.out.println();
- }
- System.out.println();
- }
- public static void main(String[] args) {
- int startX = 0;
- int startY = 0;
- DfsRatMaze d = new DfsRatMaze();
- d.initMaze();
- d.dfs(startX, startY, 0);
- if(d.min < Integer.MAX_VALUE)
- System.out.println("最少需要" + d.min + "步");
- else
- System.out.println("目标地点无法到达");
- }
- }
运行后输出:
- [1,0]->[1,1]->[2,1]->[3,1]->[4,1]->[4,2]->[4,3]->[3,3]
- [1,0]->[1,1]->[2,1]->[3,1]->[3,0]->[4,0]->[4,1]->[4,2]->[4,3]->[3,3]
- [1,0]->[1,1]->[0,1]->[0,2]->[0,3]->[1,3]->[2,3]->[3,3]
- [0,1]->[1,1]->[2,1]->[3,1]->[4,1]->[4,2]->[4,3]->[3,3]
- [0,1]->[1,1]->[2,1]->[3,1]->[3,0]->[4,0]->[4,1]->[4,2]->[4,3]->[3,3]
- [0,1]->[0,2]->[0,3]->[1,3]->[2,3]->[3,3]
- 最少需要6步
可以看到,程序计算出了所有路线,并找到了最短的路线。而整个代码还不到100行,真是神奇的算法。
Java与算法之(5) - 老鼠走迷宫(深度优先算法)的更多相关文章
- 老鼠走迷宫I
转自:http://blog.csdn.net/holymaple/article/details/8582517 说明:老鼠走迷宫是递回求解的基本提醒,我们在二维阵列中使用2来表示迷宫墙壁,使用1来 ...
- 老鼠走迷宫II
转自:http://blog.csdn.net/holymaple/article/details/8636234 由于迷宫的设计,老鼠走迷宫的入口至出口路径可能不止一条,如何求出所有的路径呢? 解法 ...
- noj->电子老鼠走迷宫
00 问题 描述: 有一只电子老鼠被困在如下图所示的迷宫中.这是一个12*12单元的正方形迷宫,黑色部分表示建筑物,白色部分是路.电子老鼠可以在路上向上.下.左.右行走,每一步走一个格子.现给定一个起 ...
- noj电子老鼠走迷宫(深搜dfs)超时错误
1042.电子老鼠闯迷宫 时限:1000ms 内存限制:10000K 总时限:3000ms 描述 有一只电子老鼠被困在如下图所示的迷宫中.这是一个12*12单元的正方形迷宫,黑色部分表示建筑物,白色 ...
- 老鼠走迷宫(2)输出所有路径(C语言)
需求 有一个迷宫,在迷宫的某个出口放着一块奶酪.将一只老鼠由某个入口处放进去,它必须穿过迷宫,找到奶酪.请找出它的行走路径. STEP 1 题目转化 我们用一个二维数组来表示迷宫,用2表示迷宫的墙壁, ...
- 老鼠走迷宫(1)输出唯一路径(C语言)
需求 有一个迷宫,在迷宫的某个出口放着一块奶酪.将一只老鼠由某个入口处放进去,它必须穿过迷宫,找到奶酪.请找出它的行走路径. STEP 1 题目转化 我们用一个二维数组来表示迷宫,用2表示迷宫的墙壁, ...
- Java与算法之(12) - 老鼠再闯迷宫(广度优先算法)
贪吃的小老鼠又回来了,这次有什么新的办法吃到奶酪呢? 规则不变,只能上下左右在格子内移动. 因为上次的深度优先算法让老鼠走了不少冤枉路,这次老鼠带来了帮手探路鼠.探路鼠的使用规则如下: 小老鼠按右.下 ...
- Java基于OpenCV实现走迷宫(图片+路线展示)
Java基于OpenCV实现走迷宫(图片+路线展示) 由于疫情,待在家中,太过无聊.同学发了我张迷宫图片,让我走迷宫来缓解暴躁,于是乎就码了一个程序出来.特此记录. 原图: 这张图,由于不是非常清晰, ...
- golang 实现广度优先算法(走迷宫)
maze.go package main import ( "fmt" "os" ) /** * 广度优先算法 */ /** * 从文件中读取数据 */ fun ...
随机推荐
- .NET 对 XML 进行创建,增加,删除,修改操作整理
前言: 最近做了一个项目,程序A在一个服务器程序B在另一台服务器,然而主程序A需要访问程序B的图片集文件夹下载到本服务器上,为了防止多次对Web Services进行调用,在主程序A中创建一个XML文 ...
- Spring Tool Suite生成默认的MVC项目的配置文件问题
1.STS是开发Spring程序的首选,基于JavaEE的程序,我都用STS来开发,但是在生成默认的MVC项目时,其配置文件让人很讨厌,在许多选项上都会加一个beans,如<beans:bean ...
- 数据分析与展示——Matplotlib基础绘图函数示例
Matplotlib库入门 Matplotlib基础绘图函数示例 pyplot基础图表函数概述 函数 说明 plt.plot(x,y,fmt, ...) 绘制一个坐标图 plt.boxplot(dat ...
- Kafka详细的设计和生态系统
欢迎大家前往云加社区,获取更多腾讯海量技术实践干货哦~ 译者:人工智能资讯小编 本译文自Jean-Paul Azar 在 https://dzone.com 发表的 Kafka Detailed De ...
- bzoj 2427: [HAOI2010]软件安装
Description 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得这些软件的价值尽可能大(即Vi的和 ...
- 开发过程遇到的bug
1.transition和display的冲突: 解决方法:1)使用visibility:2)使用js的setTimeout使动画延迟:3)可以改变高度: 2.form表单提交和js中的提交冲突 解决 ...
- df 命令详解
一.df 作用: 显示磁盘分区上的可使用的磁盘空间, 默认显示单位为kb . 可以利用该命令来获取硬盘被占用了多少空间,目前还剩下多少空间的等信息. 选项: -a :包含全部的文件系统 -h :以 ...
- 详解PHP反射API
PHP中的反射API就像Java中的java.lang.reflect包一样.它由一系列可以分析属性.方法和类的内置类组成.它在某些方面和对象函数相似,比如get_class_vars(),但是更加灵 ...
- Illustration of Git branching and merge
网上看到的描述Git工作流程的图片,有些出处忘了保存,仅供学习. 1. One Git Branching Model 出处: http://nvie.com/posts/a-successful-g ...
- 不解释,分享这个base.css
@charset "utf-8"; /*! * @名称:base.css * @功能:1.重设浏览器默认样式 * 2.设置通用原子类 */ /* 防止用户自定义背景颜色对网页的影响 ...